首页 > 数据库 >如何防止PHP中的SQL注入?

如何防止PHP中的SQL注入?

时间:2023-11-04 23:49:00浏览次数:44  
标签:语句 name PDO SQL PHP 预处理 注入

内容来自 DOC https://q.houxu6.top/?s=如何防止PHP中的SQL注入?

如果没有对用户输入进行任何修改就插入到SQL查询中,那么应用程序就会容易受到SQL注入攻击,就像以下示例中的那样:

$unsafe\_variable = $\_POST['user\_input']; 

mysql\_query("INSERT INTO `table` (`column`) VALUES ('$unsafe\_variable')");

这是因为用户可以输入类似于“value'); DROP TABLE table;--”的内容,查询就变成了:

INSERT INTO `table` (`column`) VALUES('value'); DROP TABLE table;--')

为了防止这种情况发生,可以采取以下措施:

  1. 使用预处理语句(Prepared Statements)或参数化查询(Parameterized Queries),这样可以将用户输入作为参数传递给查询,而不是将其直接嵌入到查询中。例如:
$unsafe_variable = $_POST['user_input']; 

mysql_query("INSERT INTO `table` (`column`) VALUES (?)", $unsafe_variable);
  1. 对用户输入进行验证和过滤,确保输入的数据符合预期的格式和类型。可以使用正则表达式或其他验证方法来实现。例如:
$unsafe_variable = $_POST['user_input']; 

if (preg_match('/^[a-zA-Z0-9]+$/', $unsafe_variable)) {
    mysql_query("INSERT INTO `table` (`column`) VALUES (?)", $unsafe_variable);
} else {
    // 处理无效输入的情况
}

避免SQL注入攻击的正确方法是将数据与SQL分离,这样数据将保持数据的形式,并且永远不会被SQL解析器解释为命令。无论使用哪种数据库,都可以通过使用预处理语句和参数化查询来确保安全。这些是单独发送到数据库服务器并与任何参数一起解析的SQL语句。这样,攻击者就无法注入恶意SQL。

基本上有两种方法来实现这一点:

  1. 对于支持的数据库驱动程序,可以使用PDO
$stmt = $pdo->prepare('SELECT * FROM employees WHERE name = :name');
$stmt->execute(['name' => $name]);

foreach ($stmt as $row) {
    // 使用$row做一些事情
}
  1. 对于MySQL,可以使用MySQLi(自PHP 8.2+):

自PHP 8.2+起,我们可以利用execute_query()**方法,它将准备、绑定参数和执行SQL语句结合在一起:

$result = $db->execute_query('SELECT * FROM employees WHERE name = ?', [$name]);
while ($row = $result->fetch_assoc()) {
    // 使用$row做一些事情
}

直到PHP 8.1:

$stmt = $db->prepare('SELECT * FROM employees WHERE name = ?');
$stmt->bind_param('s', $name); // 's'指定变量类型 => 'string'
$stmt->execute();
$result = $stmt->get_result();
while ($row = $result->fetch_assoc()) {
    // 使用$row做一些事情
}

如果连接到的数据库不是MySQL,还可以参考特定于驱动程序的第二个选项(例如,pg_prepare()pg_execute()用于PostgreSQL)。PDO是通用选项。

正确设置连接

PDO

注意,当使用PDO访问MySQL数据库*时,默认情况下不会使用真正的预处理语句。要修复此问题,必须禁用模拟预处理语句。以下是使用PDO创建连接的示例:

$dbConnection = new PDO('mysql:dbname=dbtest;host=127.0.0.1;charset=utf8mb4', 'user', 'password');

$dbConnection->setAttribute(PDO::ATTR_EMULATE_PREPARES, false);
$dbConnection->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION);

在上面的示例中,错误模式并非严格必要,但建议添加它。这样,PDO将通过抛出PDOException通知您所有MySQL错误。

然而,强制要求的是第一行setAttribute(),它告诉PDO禁用模拟预处理语句并使用真正的预处理语句。这可以确保在将语句和值发送到MySQL服务器之前,它们不会被PHP解析(从而给攻击者提供注入恶意SQL的机会)。

尽管可以在构造函数的选项中设置charset,但值得注意的是,在旧版本的PHP(低于5.3.6)中,DSN中的charset参数会被静默忽略。

Mysqli

对于mysqli,我们必须遵循相同的例程:

mysqli_report(MYSQLI_REPORT_ERROR | MYSQLI_REPORT_STRICT); // 错误报告
$dbConnection = new mysqli('127.0.0.1', 'username', 'password', 'test');
$dbConnection->set_charset('utf8mb4'); // 字符集

重要的是参数值与编译后的语句结合在一起,而不是一个SQL字符串。SQL注入通过在创建要发送到数据库的SQL时欺骗脚本来包含恶意字符串。因此,通过将实际的SQL与参数分开发送,可以限制最终得到意外结果的风险。

在使用预处理语句时发送的任何参数都将被视为字符串(尽管数据库引擎可能会进行一些优化,使参数也可能成为数字)。在上面的示例中,如果$name变量包含'Sarah'; DELETE FROM employees,结果将只是搜索字符串"'Sarah'; DELETE FROM employees",您不会最终得到一个空表(http://xkcd.com/327/)。

使用预处理语句的另一个好处是,如果您在同一会话中多次执行相同的语句,它只会被解析和编译一次,为您提供一些速度提升。

哦,既然您问到如何为插入操作做这件事,这里有一个示例(使用PDO):

$preparedStatement = $db->prepare('INSERT INTO table (column) VALUES (:column)');

$preparedStatement->execute([ 'column' => $unsafeValue ]);

预处理语句可用于动态查询吗?

虽然您可以继续使用预处理语句作为查询参数,但动态查询的结构本身无法参数化,某些查询功能也无法参数化。

对于这些特定场景,最好的做法是使用白名单过滤器来限制可能的值。

// 值白名单
// $dir只能为'DESC',否则为'ASC'
if (empty($dir) || $dir !== 'DESC') {
   $dir = 'ASC';
}

标签:语句,name,PDO,SQL,PHP,预处理,注入
From: https://www.cnblogs.com/xiaomandujia/p/17810048.html

相关文章

  • NodeJS系列(13)- Next.js 框架 (六) | Node.js + Next.js + Prisma/Sequelize (ORM) + M
    Next.js是一个用于构建Web应用程序的框架。Next.js是一个用于生产环境的React框架,是一个React服务端渲染应用框架。NextJS:https://nextjs.org/Prisma是一个基于promise的Node.js和TypeScript的ORM,目前支持Mysql,MariaDB,SQLite,PostgreSQL,AWSAuroraServerles......
  • mysql 8.2 安装和赋权
    1-下载安装包下载地址: https://dev.mysql.com/downloads/mysql/ 2-安装制定目录安装,略。3-登录,创建用户并赋权1C:\Users\Administrator>mysql-uroot-p2Enterpassword:******3WelcometotheMySQLmonitor.Commandsendwith;or\g.4YourMySQL......
  • MariaDB(MySQL)的常用命令3 【使用通配符过滤】
    第8章使用通配符过滤LIKE操作符百分号(%)通配符(匹配多个字符,类似?)SELECT*FROMstudentsWHEREemailLIKE'%@163.com';-下划线(_)通配符(匹配单个字符,类似*)SELECT*FROMstudentsWHEREnameLIKE'张_';Tips:1.查找的字符串,可能是大小......
  • mysql相关知识
    1、cmd连接mysql命令mysql-h192.168.0.200-u用户名-p回车后输入密码2、创建数据库createdatabasegx_day15DEFAULTCHARSETutf8COLLATEutf8_general_ci;3、django连接数据库DATABASES={'default':{'ENGINE':'django.db.backends.mysql','NAME�......
  • C/C++连接mysql(api接口方法详解)
      前言本篇记录C/C++连接mysql利用mysql的api接口的方法:这个方法的代码基本上很久都没有变过了,这里做个笔记来简单学习一下,还有一种方法等有时间了解后再来更新使用API的方式连接,需要先做环境配置,加载mysql的头文件和lib文件。可以看我之前的一篇文章VS中C/C++访问MySQL数据......
  • MySQL正则表达式
    1. 正则表达式运算符expressionNOTREGEXPpattern,expressionNOTRLIKEpatternexpressionREGEXPpattern,expressionRLIKEpattern如果表达式expression匹配给定模式pattern返回1,否则返回0.如果表达式expression或者模式pattern为NULL,则返回NULLRLIKE和REGEXP一样.2......
  • MySQL 获取MySQL列中字符串出现的次数
    使用SUM()和LIKE语句计算字符串出现次数首先,我们可以使用SUM()函数和LIKE语句计算特定字符串在某一列中出现的次数。具体实现方法如下:SELECTSUM(CASEWHENcolumn_nameLIKE'%search_string%'THEN1ELSE0END)assearch_countFROMtable_name;SQLCopy其中,column_name为需要......
  • Mysql查询字符串中某个字符串出现的次数
    目录1.查单个字符出现的次数2.查多个字符出现的次数3.函数讲解1.查单个字符出现的次数比如我想查howdoyoudo字符串当中出现d的次数:第一眼看上去有点懵,首先mysql并没有直接计算出现字符次数的函数,所以才使用了下面这种方式,其实就是将出现的字符串给替换为了空。然后让原数据减去......
  • doltgresql dolthub 团队开源的基于pg 协议的数据版本化管理工具
    dolthub团队开源了几个比较重要的东西(dolt)以及持续维护了go-mysql-server,dolt是基于go-mysql-server的数据多版本管理工具,从官方介绍上doltgresql的架构与dolthub是类似的dolthub参考架构说明dolthub基于git模式的数据管理是很值得学习的,是一个很不错数据多版本管理工......
  • SqlServer各年份版本
    一、年代版本8-SQLServer20009-SQLServer200510-SQLServer200811-SQLServer201212-SQLServer201413-SQLServer201614-SQLServer201715-SQLServer201916-SQLServer2022二、开发版、标准版、企业版版本差异1、SQLServer2016......