0

0

PHP实现DevExtreme过滤条件到MySQL WHERE子句的转换

碧海醫心

碧海醫心

发布时间:2025-09-23 17:17:01

|

955人浏览过

|

来源于php中文网

原创

PHP实现DevExtreme过滤条件到MySQL WHERE子句的转换

本文详细介绍了如何使用PHP将DevExtreme前端框架生成的类NoSQL过滤条件数组,安全有效地转换为MySQL数据库的WHERE子句。文章提供了针对PDO和MySQLi两种数据库扩展的实现方案,包括生成带参数占位符的SQL语句和提取对应参数值的函数,旨在帮助开发者构建健壮的数据查询接口。

在现代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进行转换

使用PDO(PHP Data Objects)是PHP连接数据库的推荐方式,因为它支持预处理语句,能够有效防止SQL注入攻击。我们需要编写两个辅助函数:一个用于生成带有参数占位符的SQL查询字符串,另一个用于从过滤数组中提取这些参数的值。

立即学习PHP免费学习笔记(深入)”;

假设我们有以下DevExtreme风格的过滤数组:

酷兔AI论文
酷兔AI论文

专业原创高质量、低查重,免费论文大纲,在线AI生成原创论文,AI辅助生成论文的神器!

下载
$filterArray = [["SizeCd","=","UNIT"],"or",["SizeCd","=","JOGO"],"or",["SizeCd","=","PACOTE"]];

1. 生成带参数占位符的SQL查询字符串

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;
}

2. 提取参数值数组

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使用示例

结合这两个函数,我们可以构建并执行安全的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进行转换(非预处理方式)

如果项目仍在使用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();

注意事项与总结

  1. 安全性
    • 强烈推荐使用PDO的预处理语句。它通过将SQL逻辑与数据分离,是防御SQL注入最有效的方法。arrayToQuery 和 arrayToParams 组合正是为PDO预处理设计的。
    • 如果必须使用MySQLi且不采用预处理,务必使用 mysqli_real_escape_string() 函数对所有用户输入的值进行转义,以防止恶意字符破坏SQL语句结构。
  2. 字段名引用:在生成的SQL中,字段名(如 SizeCd)使用反引号(`)括起来。这是一种良好的实践,可以避免字段名与SQL关键字冲突。
  3. 灵活性与扩展性
    • 当前的实现适用于简单的 AND/OR 条件链。对于更复杂的嵌套过滤(例如 (A AND B) OR C),可能需要更高级的解析逻辑,例如递归函数来处理嵌套数组。
    • 如果需要支持更多的运算符(如 LIKE, IN, BETWEEN),则需要扩展 arrayToQuery 函数的逻辑来识别并正确处理这些运算符及其对应的SQL语法。
  4. 错误处理:在实际应用中,应始终包含适当的错误处理机制,例如 try-catch 块处理PDO异常,或检查MySQLi查询结果。
  5. 性能:对于非常复杂的过滤条件,生成SQL字符串可能会有轻微的性能开销。但对于大多数应用场景,这种开销可以忽略不计。

通过上述PHP函数,开发者可以高效且安全地将DevExtreme等前端框架的类NoSQL过滤条件转换为MySQL的 WHERE 子句,从而实现前后端数据交互的无缝对接。

相关专题

更多
php文件怎么打开
php文件怎么打开

打开php文件步骤:1、选择文本编辑器;2、在选择的文本编辑器中,创建一个新的文件,并将其保存为.php文件;3、在创建的PHP文件中,编写PHP代码;4、要在本地计算机上运行PHP文件,需要设置一个服务器环境;5、安装服务器环境后,需要将PHP文件放入服务器目录中;6、一旦将PHP文件放入服务器目录中,就可以通过浏览器来运行它。

2735

2023.09.01

php怎么取出数组的前几个元素
php怎么取出数组的前几个元素

取出php数组的前几个元素的方法有使用array_slice()函数、使用array_splice()函数、使用循环遍历、使用array_slice()函数和array_values()函数等。本专题为大家提供php数组相关的文章、下载、课程内容,供大家免费下载体验。

1669

2023.10.11

php反序列化失败怎么办
php反序列化失败怎么办

php反序列化失败的解决办法检查序列化数据。检查类定义、检查错误日志、更新PHP版本和应用安全措施等。本专题为大家提供php反序列化相关的文章、下载、课程内容,供大家免费下载体验。

1530

2023.10.11

php怎么连接mssql数据库
php怎么连接mssql数据库

连接方法:1、通过mssql_系列函数;2、通过sqlsrv_系列函数;3、通过odbc方式连接;4、通过PDO方式;5、通过COM方式连接。想了解php怎么连接mssql数据库的详细内容,可以访问下面的文章。

975

2023.10.23

php连接mssql数据库的方法
php连接mssql数据库的方法

php连接mssql数据库的方法有使用PHP的MSSQL扩展、使用PDO等。想了解更多php连接mssql数据库相关内容,可以阅读本专题下面的文章。

1444

2023.10.23

html怎么上传
html怎么上传

html通过使用HTML表单、JavaScript和PHP上传。更多关于html的问题详细请看本专题下面的文章。php中文网欢迎大家前来学习。

1235

2023.11.03

PHP出现乱码怎么解决
PHP出现乱码怎么解决

PHP出现乱码可以通过修改PHP文件头部的字符编码设置、检查PHP文件的编码格式、检查数据库连接设置和检查HTML页面的字符编码设置来解决。更多关于php乱码的问题详情请看本专题下面的文章。php中文网欢迎大家前来学习。

1549

2023.11.09

php文件怎么在手机上打开
php文件怎么在手机上打开

php文件在手机上打开需要在手机上搭建一个能够运行php的服务器环境,并将php文件上传到服务器上。再在手机上的浏览器中输入服务器的IP地址或域名,加上php文件的路径,即可打开php文件并查看其内容。更多关于php相关问题,详情请看本专题下面的文章。php中文网欢迎大家前来学习。

1307

2023.11.13

Java编译相关教程合集
Java编译相关教程合集

本专题整合了Java编译相关教程,阅读专题下面的文章了解更多详细内容。

9

2026.01.21

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
MySQL 教程
MySQL 教程

共48课时 | 1.8万人学习

MySQL 初学入门(mosh老师)
MySQL 初学入门(mosh老师)

共3课时 | 0.3万人学习

简单聊聊mysql8与网络通信
简单聊聊mysql8与网络通信

共1课时 | 805人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号