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