预准备语句是一种用于高效重复执行相同(或相似)SQL 语句的功能。
mysqli_prepare() 函数用于准备语句。准备好的语句是一个 SQL 语句模板,其中某些未指定的值称为参数(标记为"?")。例如:INSERT INTO Employee VALUES (?, ?, ?)。
数据库对SQL语句模板进行解析、编译、查询优化,并存储结果而不执行。稍后,应用程序在执行语句之前使用 mysqli_stmt_bind_param() 函数将值绑定到参数。应用程序可以使用不同的值多次执行该语句。
注意:标记 (?) 仅在某些情况下才合法SQL 语句中的位置。例如,它们可以出现在 INSERT 语句的 VALUES() 列表中(以指定行的列值),或者可以出现在与 WHERE 子句中的列的比较中以指定比较值。
但是,它们不允许用于标识符(例如表名或列名),也不允许指定二元运算符的两个操作数(例如 = 等号)。后一个限制是必要的,因为无法确定参数类型。
一般来说,参数仅在数据操作语言(DML)语句中合法,而在数据定义语言(DDL)语句中不合法。与直接执行 SQL 语句相比,预准备语句具有三个主要优点:
- 准备好的语句减少了解析时间,因为查询的准备工作只需完成一次(尽管该语句被执行多次)。
- 绑定参数最大限度地减少了服务器的带宽,因为您每次只需要发送参数,而不是整个查询。
- 准备好的语句对于防止 SQL 注入非常有用,因为稍后使用不同协议传输的参数值不需要正确转义。如果原始语句模板不是来自外部输入,则不会发生 SQL 注入。
准备语句 - 面向对象风格
下面的示例演示了如何准备语句使用面向对象的风格。
<?php
//建立与数据库的连接
$mysqli = new mysqli("localhost", "user", "password", "database");
if ($mysqli->connect_errno) {
echo "Failed to connect to MySQL: ". $mysqli->connect_error;
exit();
}
//准备执行SQL语句
$query = "INSERT INTO Employee (Name, City, Salary) VALUES (?, ?, ?)";
$stmt = $mysqli->prepare($query);
//绑定参数
$stmt->bind_param('ssd', $name, $city, $salary);
//设置参数并执行
$name = "John";
$city = "London";
$salary = 2800;
$stmt->execute();
$name = "Marry";
$city = "Paris";
$salary = 2850;
$stmt->execute();
echo "Records inserted successfully.";
//关闭连接
$mysqli->close();
?>
上述代码的输出将是:
Records inserted successfully.
准备语句 - 面向过程风格
使用面向过程也可以实现相同的效果使用以下脚本的样式:
<?php
//建立与数据库的连接
$mysqli = mysqli_connect("localhost", "user", "password", "database");
if (mysqli_connect_errno()) {
echo "Failed to connect to MySQL: ". mysqli_connect_error();
exit();
}
//准备执行SQL语句
$query = "INSERT INTO Employee (Name, City, Salary) VALUES (?, ?, ?)";
$stmt = mysqli_prepare($mysqli, $query);
//绑定参数
mysqli_stmt_bind_param($stmt, 'ssd', $name, $city, $salary);
//设置参数并执行
$name = "John";
$city = "London";
$salary = 2800;
mysqli_stmt_execute($stmt);
$name = "Marry";
$city = "Paris";
$salary = 2850;
mysqli_stmt_execute($stmt);
echo "Records inserted successfully.";
//关闭连接
mysqli_close($mysqli);
?>
上述代码的输出将是:
Records inserted successfully.
完整 PHP MySQLi 参考
所有属性、方法的完整参考PHP MySQLi 扩展的功能和功能请参见 PHP MySQLi 参考。