
在现代web应用开发中,前端框架如devextreme常以一种结构化的数组形式定义数据过滤条件,这种格式类似于nosql查询语法,但后端数据库通常是关系型数据库如mysql。将前端的过滤条件动态转换为后端可执行的sql where 子句,是构建灵活数据接口的关键一环。
考虑以下MySQL表结构:
| item_id | StockNo | SizeCd |
|---|---|---|
| 1 | 12003 | UNIT |
| 2 | 12007 | JOGO |
| 3 | 12008 | PACOTE |
| 4 | 12033 | JOGO |
| 5 | 12034 | JOGO |
| 6 | 12038 | UNIT |
前端DevExtreme框架可能发送如下的过滤请求体:
{
"from": "get_data",
"skip": 0,
"take": 50,
"requireTotalCount": true,
"filter": [["SizeCd","=","UNIT"],"or",["SizeCd","=","JOGO"]]
}其中,filter 字段是一个数组,它定义了查询的条件。我们的目标是将这个数组转换为符合MySQL语法的 WHERE 子句,例如:WHERESizeCd= 'UNIT' ORSizeCd= 'JOGO'。在转换过程中,需要特别注意字段名不加引号,而字符串值需要加引号,并确保防止SQL注入。
使用PDO(PHP Data Objects)是PHP连接数据库的推荐方式,因为它支持预处理语句,能够有效防止SQL注入攻击。我们需要编写两个辅助函数:一个用于生成带有参数占位符的SQL查询字符串,另一个用于从过滤数组中提取这些参数的值。
立即学习“PHP免费学习笔记(深入)”;
假设我们有以下DevExtreme风格的过滤数组:
$filterArray = [["SizeCd","=","UNIT"],"or",["SizeCd","=","JOGO"],"or",["SizeCd","=","PACOTE"]];
arrayToQuery 函数负责遍历过滤数组,构建一个包含问号占位符(?)的SQL WHERE 子句。
/**
* 将DevExtreme风格的过滤数组转换为带占位符的SQL WHERE子句。
*
* @param string $tableName 表名。
* @param array $filterArray 过滤条件数组。
* @return string 生成的SQL查询字符串。
*/
function arrayToQuery(string $tableName, array $filterArray) : string
{
$select = "SELECT * FROM `{$tableName}` WHERE ";
foreach($filterArray as $item) {
if(is_array($item)) {
// 处理形如 ["SizeCd","=","UNIT"] 的条件
// 字段名用反引号括起来,防止与SQL关键字冲突
$select .= "`{$item[0]}` {$item[1]} ?";
} else {
// 处理形如 "or" 的逻辑运算符
$select .= " {$item} ";
}
}
return $select;
}arrayToParams 函数用于从过滤数组中提取所有需要绑定到SQL查询中的值。这些值将作为预处理语句的参数。
/**
* 从DevExtreme风格的过滤数组中提取所有参数值。
*
* @param array $filterArray 过滤条件数组。
* @return array 提取出的参数值数组。
*/
function arrayToParams(array $filterArray) : array
{
$return = [];
foreach($filterArray as $item) {
if(is_array($item)) {
// 提取条件数组中的第三个元素作为参数值
$return[] = $item[2];
}
}
return $return;
}结合这两个函数,我们可以构建并执行安全的PDO查询:
// 假设的过滤数组
$filterArray = [["SizeCd","=","UNIT"],"or",["SizeCd","=","JOGO"],"or",["SizeCd","=","PACOTE"]];
// 示例输出
var_dump(
arrayToQuery("your_table_name", $filterArray),
arrayToParams($filterArray)
);
/*
输出结果:
string(66) "SELECT * FROM `your_table_name` WHERE `SizeCd` = ? or `SizeCd` = ? or `SizeCd` = ?"
array(3) {
[0]=>
string(4) "UNIT"
[1]=>
string(4) "JOGO"
[2]=>
string(6) "PACOTE"
}
*/
// 实际PDO数据库操作
try {
// 假设 $conn 是一个已建立的PDO连接对象
$dsn = 'mysql:host=localhost;dbname=testdb;charset=utf8';
$username = 'root';
$password = 'password';
$conn = new PDO($dsn, $username, $password);
$conn->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION);
$tableName = "your_table_name"; // 替换为你的表名
$sql = arrayToQuery($tableName, $filterArray);
$params = arrayToParams($filterArray);
$stmt = $conn->prepare($sql);
$stmt->execute($params);
$results = $stmt->fetchAll(PDO::FETCH_ASSOC);
print_r($results);
} catch (PDOException $e) {
echo "数据库错误: " . $e->getMessage();
}如果项目仍在使用MySQLi扩展且不方便使用预处理语句(尽管强烈推荐使用),则需要在构建SQL字符串时手动对值进行转义,以防止SQL注入。
/**
* 将DevExtreme风格的过滤数组转换为MySQLi风格的SQL WHERE子句。
* 注意:此方法直接将值嵌入SQL,需使用mysqli_real_escape_string进行转义以防SQL注入。
*
* @param mysqli $mysqli MySQLi连接对象。
* @param string $table 表名。
* @param array $filterArray 过滤条件数组。
* @return string 生成的SQL查询字符串。
*/
function arrayToQueryMysqli($mysqli, string $table, array $filterArray) : string
{
$select = "SELECT * FROM `{$table}` WHERE ";
foreach($filterArray as $item) {
if(is_array($item)) {
// 对值进行转义并用单引号括起来
$escapedValue = $mysqli->real_escape_string($item[2]);
$select .= "`{$item[0]}` {$item[1]} '" . $escapedValue . "'";
} else {
$select .= " {$item} ";
}
}
return $select;
}
// MySQLi使用示例
// 假设 $mysqli 是一个已建立的MySQLi连接对象
$mysqli = new mysqli("localhost", "root", "password", "testdb");
if ($mysqli->connect_error) {
die("连接失败: " . $mysqli->connect_error);
}
$filterArray = [["SizeCd","=","UNIT"],"or",["SizeCd","=","JOGO"],"or",["SizeCd","=","PACOTE"]];
$tableName = "your_table_name"; // 替换为你的表名
$query = arrayToQueryMysqli($mysqli, $tableName, $filterArray);
echo "生成的SQL: " . $query . "\n";
$result = $mysqli->query($query);
if ($result) {
while ($row = $result->fetch_assoc()) {
print_r($row);
}
$result->free();
} else {
echo "查询失败: " . $mysqli->error;
}
$mysqli->close();通过上述PHP函数,开发者可以高效且安全地将DevExtreme等前端框架的类NoSQL过滤条件转换为MySQL的 WHERE 子句,从而实现前后端数据交互的无缝对接。
以上就是PHP实现DevExtreme过滤条件到MySQL WHERE子句的转换的详细内容,更多请关注php中文网其它相关文章!
PHP怎么学习?PHP怎么入门?PHP在哪学?PHP怎么学才快?不用担心,这里为大家提供了PHP速学教程(入门到精通),有需要的小伙伴保存下载就能学习啦!
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号