“参数号无效:未定义参数”插入数据

时间:2011-05-03 19:07:53

标签: php mysql parameters yii

更新

我在列出VALUES时犯了一个小错误。我应该输入“:username”而不是“:alias”。我想这个问题的答案可归功于任何想要它的人的自由统治吗?或者我是否删除了这个问题?

ORIGINAL

我一直在使用Yii的活跃记录模式。现在,我的项目需要为一个小事务访问不同的数据库。我认为Yii的DAO对此有好处。但是,我收到了一个神秘的错误。

  

CDbCommand无法执行SQL语句:SQLSTATE [HY093]:参数号无效:参数未定义

这是我的代码:

public function actionConfirmation
{
    $model_person = new TempPerson();

    $model = $model_person->find('alias=:alias',array(':alias'=>$_GET['alias']));
    $connection=Yii::app()->db2;
            $sql = "INSERT INTO users (username, password, ssn, surname
                    , firstname, email, city, country) 
                    VALUES(:alias, :password, :ssn, :surname
                    , :firstname, :email, :city, :country)";
            $command=$connection->createCommand($sql);
            $command->bindValue(":username", $model->alias);
            $command->bindValue(":password", substr($model->ssn, -4,4));
            $command->bindValue(":ssn", $model->ssn);
            $command->bindValue(":surname", $model->lastName);
            $command->bindValue(":firstname", $model->firstName);
            $command->bindValue(":email", $model->email);
            $command->bindValue(":city", $model->placeOfBirth);
            $command->bindValue(":country", $model->placeOfBirth);
            $command->execute();
            $this->render('confirmation',array('model'=>$model));
}

这构造了以下查询(如应用程序日志中所示):

INSERT INTO users (username, password, ssn, surname, firstname, email
                   , city, country) 
VALUES(:alias, :password, :ssn, :surname, :firstname, :email, :city, :country);

FYI $model->placeOfBirth应该是城市和县的价值观。这不是一个错字(我只是一件傻事)。

4 个答案:

答案 0 :(得分:87)

只是提供一个答案 - 因为这个错误很常见 - 这里有几个原因:

1):parameter名称与错误绑定不匹配(拼写错误?)。这就是这里发生的事情。他在SQL语句中有:alias但绑定:username。因此,当尝试param绑定时,Yii / PDO在sql语句中找不到:username,这意味着它是“一个参数短”并且引发了错误。

2)完全忘记为参数添加bindValue()。在Yii等$critera构造中更容易做到这一点,你有一个数组或参数($criteria->params = array(':bind1'=>'test', ':bind2'=>'test))。

3)使用togetherjoins时与CDataProvider分页和/或排序的奇怪冲突。没有特定的,简单的方法来表征这一点,但是当在CDataProviders中使用复杂查询时,我遇到了一些奇怪的问题,即参数被丢弃并且发生了这种错误。

在Yii中解决这些问题的一个非常有用的方法是在配置文件中enable parameter logging。将其添加到配置文件中的db数组中:

'enableParamLogging'=>true,

并确保在CWebLogRoute部分设置log路线。这将打印出给出的查询和错误,以及它试图绑定的所有参数。超级有帮助!

答案 1 :(得分:4)

可能是你试图用单引号绑定一个参数而不是让它为你工作。

比较

Model::model()->findAll("t.description ilike '%:filter%'", array(':filter' => $filter));

使用:

Model::model()->findAll("t.description ilike :filter", array(':filter' => '%' . $filter . '%'));

答案 2 :(得分:4)

上面没有提到的这个错误的原因是当你处理一个动态的参数数组时,如果你取消设置任何参数,你需要在传入它们之前重新索引。这个的残酷部分是你的错误日志不显示索引所以看起来一切都是正确的。例如:

SELECT id WHERE x = ?, y = ?, z = ?

可能产生Log:无效的参数号:参数未定义为params(“x”,“y”,“z”)

这看起来不应该抛出错误,但如果索引类似于:

0 => x, 1 => y, 4 => z

它认为最后一个参数未定义,因为它正在寻找键2。

答案 3 :(得分:2)

尝试执行以下操作时出现此错误:

$stmt = $pdo->prepare("select name from mytable where id = :id");
$stmt->execute([
  'id' => $id,
  'unusedvar' => $foo, // This row causes the error.
]);

基本上,您在传递给execute()的数组中没有未使用的参数。传递给execute()的数组中的每个值都必须在准备好的语句中使用。

这也在docs

中指定
  

无法绑定多于指定值的值;如果input_parameters中存在的键多于PDO :: prepare()中指定的SQL,则语句将失败并发出错误。