PDO:多次使用同一绑定变量


PDO: Using same bind variable multiple times

我对PDO Prepared语句有问题,如果需要多次使用同一绑定变量,查询将无法验证。

示例:

$params = array (
    ':status' => $status,
    ':userid' => $_SESSION['userid']
);
$stmt = $pdo->prepare ('
    INSERT INTO 
        tableName
        ( userId, status )
        VALUES
        ( :userid, ":status" )
        ON DUPLICATE KEY UPDATE
            status = ":status"
');
if ( ! $stmt->execute ( $params ))
{
    print_r( $stmt->errorInfo ());
}

编辑$params的值为:
Array ( [:status] => PAID [:userid] => 111 )

编辑2
我注意到,插入的不是原始值,而是userid,而是0,插入的是空字符串,而不是status。

问题是:status周围的引号。删除了引号,一切都很好。

您的数组键不需要包含冒号。冒号纯粹是为了让PDO知道下面是一个命名参数。

$params = array (
    ':status' => $status,
    ':userid' => $_SESSION['userid']
);

应该是

$params = array (
    'status' => $status,
    'userid' => $_SESSION['userid']
);

您没有在任何地方调用bindParam方法,为什么?在调用execute之前,请尝试添加

$stmt->bindParam(':userid', $_SESSION['userid'], PDO::PARAM_INT);
$stmt->bindParam(':status', $status, PDO::PARAM_STR);

然后打电话给$stmt->execute();,看看这对你有什么好处。还要将错误消息设置为满,在实例化PDO实例后,添加此$db->setAttribute(PDO::ATTR_ERRMODE,PDO::ERRMODE_EXCEPTION);以确保始终抛出错误。

文档是方便的东西

对我来说很好。例如

<?php
$pdo = new PDO('mysql:host=localhost;dbname=test;charset=utf8', 'localonly', 'localonly');
$pdo->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION);
setup($pdo);
echo 'client version: ', $pdo->getAttribute(PDO::ATTR_CLIENT_VERSION), "'n";
echo 'server version: ', $pdo->getAttribute(PDO::ATTR_SERVER_VERSION), "'n";
echo "before:'n";
foreach( $pdo->query('SELECT * FROM tmpTableName', PDO::FETCH_ASSOC) as $row ) {
    echo join(', ', $row), "'n";
}
$status = 1;
$_SESSION['userid'] = 'foo';
$params = array (
    'status' => $status,
    'userid' => $_SESSION['userid'],
);
$stmt = $pdo->prepare ('
    INSERT INTO 
        tmpTableName
        (userId, status)
    VALUES
        (:userid, :status)
    ON DUPLICATE KEY UPDATE
        status = :status
');
if ( ! $stmt->execute ( $params ))
{
    print_r( $stmt->errorInfo ());
}
echo "after:'n";
foreach( $pdo->query('SELECT * FROM tmpTableName', PDO::FETCH_ASSOC) as $row ) {
    echo join(', ', $row), "'n";
}
function setup($pdo) {
    $pdo->exec('
        CREATE TEMPORARY TABLE tmpTableName (
            userId varchar(32),
            status int,
            unique key(userId)
        )
    ');
    $pdo->exec("INSERT INTO tmpTableName (userId,status) VALUES ('foo', 0)");
    $pdo->exec("INSERT INTO tmpTableName (userId,status) VALUES ('bar', 0)");
}

打印

client version: mysqlnd 5.0.10 - 20111026 - $Id: b0b3b15c693b7f6aeb3aa66b646fee339f175e39 $
server version: 5.5.25a
before:
foo, 0
bar, 0
after:
foo, 1
bar, 0

在我的机器上