PHP-PDO参数绑定问题
前言
今天在執行這樣一段代碼:
$data = ['username' => 'hujingnb','address' => 'beijing', ]; $dbh = new PDO("mysql:host={$host};dbname={$dbname}", $username, $password); $statement = $dbh->prepare('INSERT INTO `test_user` (`username`, `address`) VALUES (:username, :address)'); foreach($data as $key => $value ){$statement->bindParam($key, $value); } $ret = $statement->execute();這意思很明顯了嘛. 但是, 當我看到插入的數據內容時, 我傻眼了.
什么? 為什么會這樣, 一共就5行代碼, 我哪里寫的有問題?
解惑
最終, 還是編譯器救了我. 我看到了這個:
注意我圈中的地方, 也就是說, 這個方法接了一個引用的值. 這樣我想起了之前遇到的一個相同的坑, 具體可看我這篇文章: PHP 循環引用的問題
簡單來說, 就是bindParam方法, 接到$value變量的地址存起來了, 每次循環拿到的都是同一個地址. 等到循環結束的時候, $value所指定的值為數組的最后一個, 此時, statement對象中記錄的所有$value自然也都是數組最后一個. 插入數據全部為’beijing’也就不奇怪了.
解決
既然知道了問題出在地址引用上, 解決的思路就放在地址的引用上了. 怎么解決呢?
1. unset
每次循環結束時, 都調用unset方法, 將$value釋放調, 這樣下次進入循環是,又是一個新的變量了.
foreach($data as $key => $value ){$statement->bindParam($key, $value);unset($value); }2. 將遍歷去掉
不使用foreach, 直接進行賦值, 這樣也就沒有中間變量的問題拉.
$statement->bindValue(":username", $data['username']); $statement->bindValue(":address", $data['address']);3.使用引用接
在foreach中, 直接用引用接$value的內容, 這樣接到的本身, 就是數據value的引用對象, 每個$value的地址也就自然不同了.
foreach($data as $key => &$value ){$statement->bindParam($key, $value); }4. 改用bindValue
和bindParams功能差不多的方法, 還有一個bindValue. 此方法接收的是值變量, 并不接收地址.
想來對于我們這種功能情況, 官方的想法本來就是用bindValue方法吧, bindParams方法暫時沒想到什么使用場景.
5.直接將參數傳給execute
execute也可以接收參數的綁定. 故, 我們也可以將參數的綁定直接傳給它, 中間不用調bind方法了.
$ret = $statement->execute($data);最后, bindParam方法, 明顯是希望 sql已經創建之后, 其中的值會延后獲取. 不過我想了想沒有想到具體的應用場景.
在最后, 對于這種參數中接了引用的方法, 在foreach中調用都要小心一些.
總結
以上是生活随笔為你收集整理的PHP-PDO参数绑定问题的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 网络带宽是什么
- 下一篇: PHP的stdClass