为什么在Java中PreparedStatement能够有效防止SQL注入?这可能是每个Java程序员思考过的问题。

首先我们来看下直观的现象(注:需要提前打开mysql的SQL文日志)

1. 不使用PreparedStatement的set方法设置参数(效果跟Statement相似,相当于执行静态SQL)

String param = "'test' or 1=1";
String sql = "select file from file where name = " + param; // 拼接SQL参数
PreparedStatement preparedStatement = connection.prepareStatement(sql);
ResultSet resultSet = preparedStatement.executeQuery();
System.out.println(resultSet.next());

输出结果为true,DB中执行的SQL为

-- 永真条件1=1成为了查询条件的一部分,可以返回所有数据,造成了SQL注入问题select file from file where name = 'test' or 1=1 

2. 使用PreparedStatement的set方法设置参数

String param = "'test' or 1=1";
String sql = "select file from file where name = ?";
PreparedStatement preparedStatement = connection.prepareStatement(sql);
preparedStatement.setString(1, param);
ResultSet resultSet = preparedStatement.executeQuery();
System.out.println(resultSet.next());

输出结果为false,DB中执行的SQL为

select file from file where name = '\'test\' or 1=1'

我们可以看到输出的SQL文是把整个参数用引号包起来,并把参数中的引号作为转义字符,从而避免了参数也作为条件的一部分


接下来我们分析下源码(以mysql驱动实现为例)

打开java.sql.PreparedStatement通用接口,看到如下注释,了解到PreparedStatement就是为了提高statement(包括SQL,存储过程等)执行的效率。

An object that represents a precompiled SQL statement.A SQL statement is precompiled and stored in a PreparedStatement object. This object can then be used to efficiently execute this statement multiple times.

那么,什么是所谓的“precompiled SQL statement”呢?

回答这个问题之前需要先了解下一个SQL文在DB中执行的具体步骤:

  1. Convert given SQL query into DB format -- 将SQL语句转化为DB形式(语法树结构)
  2. Check for syntax -- 检查语法
  3. Check for semantics -- 检查语义
  4. Prepare execution plan -- 准备执行计划(也是优化的过程,这个步骤比较重要,关系到你SQL文的效率,准备在后续文章介绍)
  5. Set the run-time values into the query -- 设置运行时的参数
  6. Run the query and fetch the output -- 执行查询并取得结果

而所谓的“precompiled SQL statement”,就是同样的SQL文(包括不同参数的),1-4步骤只在第一次执行,所以大大提高了执行效率(特别是对于需要重复执行同一SQL的)

言归正传,回到source中,我们重点关注一下setString方法(因为其它设置参数的方法诸如setInt,setDouble之类,编译器会检查参数类型,已经避免了SQL注入。)

查看mysql中实现PreparedStatement接口的类com.mysql.jdbc.PreparedStatement中的setString方法(部分代码)

    public void setString(int parameterIndex, String x) throws SQLException {synchronized (checkClosed().getConnectionMutex()) {// if the passed string is null, then set this column to nullif (x == null) {setNull(parameterIndex, Types.CHAR);} else {checkClosed();int stringLength = x.length();if (this.connection.isNoBackslashEscapesSet()) {// Scan for any nasty chars                    // 判断是否需要转义处理(比如包含引号,换行等字符)boolean needsHexEscape = isEscapeNeededForString(x, stringLength); // 如果不需要转义,则在两边加上单引号if (!needsHexEscape) {byte[] parameterAsBytes = null;StringBuilder quotedString = new StringBuilder(x.length() + 2);quotedString.append('\'');quotedString.append(x);quotedString.append('\'');...} else {...}String parameterAsString = x;boolean needsQuoted = true; // 如果需要转义,则做转义处理if (this.isLoadDataQuery || isEscapeNeededForString(x, stringLength)) {...

从上面加红色注释的可以明白为什么参数会被单引号包裹,并且类似单引号之类的特殊字符会被转义处理,就是因为这些代码的控制避免了SQL注入。

这里只对SQL注入相关的代码进行解读,如果在setString前后输出预处理语句(preparedStatement.toString()),会发现如下输出

Before bind: com.mysql.jdbc.JDBC42PreparedStatement@b1a58a3: select file from file where name = ** NOT SPECIFIED **
After bind: com.mysql.jdbc.JDBC42PreparedStatement@b1a58a3: select file from file where name = '\'test\' or 1=1'

编程中建议大家使用PrepareStatement + Bind-variable的方式避免SQL注入

大家有什么其它的看法,欢迎留下评论!

参考:https://stackoverflow.com/questions/30587736/what-is-pre-compiled-sql-statement

转载于:https://www.cnblogs.com/roostinghawk/p/9703806.html

PreparedStatement是如何防止SQL注入的?相关推荐

  1. 【转】从源码分析PreparedStatement是如何防止SQL注入的?

    为什么在Java中PreparedStatement能够有效防止SQL注入?这可能是每个Java程序员思考过的问题. 首先我们来看下直观的现象(注:需要提前打开mysql的SQL文日志) 1. 不使用 ...

  2. 2020-08-02 Mysql数据库索引初识、备份、设计原则、JDBC连接、SQL注入、PreparedStatement对象使用、事务处理、连接池

    ------------------------索引---------------------- 定义:帮助MYSQL高效获取数据的数据结构 ----------主键索引----------prima ...

  3. JDBC--代码实现增删改查、及SQL注入问题解决

    Startment.PreparedStatement对象详解及SQL注入问题 (1)Statement对象详解 1.提取工具类 #db.properties dirver=com.mysql.jdb ...

  4. 如何干掉恶心的 SQL 注入?

    点击上方蓝色"程序猿DD",选择"设为星标" 回复"资源"获取独家整理的学习资料! 来源 | rrd.me/fKXEa 简介 文章主要内容包 ...

  5. jdbc之防sql注入攻击

    1.SQL注入攻击:     由于dao中执行的SQL语句是拼接出来的,其中有一部分内容是由用户从客户端传入,所以当用户传入的数据中包含sql关键字时,就有可能通过这些关键字改变sql语句的语义,从而 ...

  6. 解决SQL注入与XSS攻击

    最近接手之前同事的几个项目,公司利用扫描工具进行全项目扫描,发现了部分项目代码存在安全漏洞,所以需要进行项目代码修复以避免有人恶意攻击.这个任务自然而然的就落到我手上.在这里记录一下操作的过程. 扫描 ...

  7. java 最新sql注入原因以及预防方案(易理解)

    前沿 在现有的框架中sql防注入已经做得很好了,我们需要做的就是尽量不要使用sql拼接调用 java sql注入原因以及预防方案(易理解) 1. SQL注入 1.1 原理 SQL注入是通过客户端的输入 ...

  8. java sql objects_Java SQL注入学习笔记

    1 简介 文章主要内容包括: Java 持久层技术/框架简单介绍 不同场景/框架下易导致 SQL 注入的写法 如何避免和修复 SQL 注入 2 JDBC 介绍 JDBC: 全称 Java Databa ...

  9. 常见的Java审计代码函数关键字_转载:Java代码审计汇总系列(一)——SQL注入

    原文链接:https://cloud.tencent.com/developer/article/1534109 一.代码审计 相比黑盒渗透的漏洞挖掘方式,代码审计具有更高的可靠性和针对性,更多的是依 ...

最新文章

  1. 专家也要小心,HTTPS网址的网站就一定安全吗?
  2. 大数据笔记2019.5.8
  3. 实践1-qq邮箱主页
  4. 进程间通信--无名管道(pipe)
  5. 【渝粤题库】国家开放大学2021春1044合同法题目
  6. 数学专业学计算机哪一行,计算数学
  7. js map对象遍历_何时使用 Map 来代替变通的 JS 对象
  8. java println源码_System.out.println()相关源码
  9. JavaScript与WebAssembly进行比较
  10. 齿轮箱常见故障数据_齿轮箱故障分析和维护使用
  11. 斜齿轮重合度计算公式_斜齿圆柱齿轮的特点?基本参数和斜齿轮的重合度是什么?...
  12. vmware 14 激活码
  13. python 内置函数_Python简介,第6章–内置函数和方法
  14. oracle box怎么全屏,Oracle VM VirtualBox 虚拟机设置全屏与共享
  15. 登录英雄联盟lol后无法显示界面出不来问题解决
  16. 用python打开文件然后写个欢迎代码
  17. 京东店铺如何批量修改主推SKU?
  18. Delphi Web前端开发教程(2):基于TMS WEB Core框架
  19. 示波器探头x10、x1挡位
  20. Android 手电筒的开启方法

热门文章

  1. Java内部类的使用小结
  2. Docker入门之五数据管理
  3. 机器学习(6): 层次聚类 hierarchical clustering
  4. Linux下最快速共享目录的方法
  5. 使用HttpClient 调用Web Api
  6. oracle xe 数据库用户操作
  7. 老男孩的学生优秀博文及内部教学文章
  8. PHP遍历文件夹下所有文件
  9. 在思科路由器上做 _限速
  10. php 实现树状组织图插件,使用jstree插件实现树形结构