首页 > 技术文章 > preparestatement可以避免注入

ctaixw 2016-07-11 21:03 原文

之所以PreparedStatement能防止注入,是因为它把单引号转义了,变成了\',这样一来,就无法截断SQL语句,进而无法拼接SQL语句,基本上没有办法注入了。 

不使用这个,我们一般做查询或更新的条件,是用字符串拼起来的,例如

1
2
String id = (String)request.getAttribute("id");    //假设页面上传了一个id值过来
String SQL = "SELECT ID,NAME FROM USER WHERE ID='" + id + "'";    //拼接成一个完整的sql语句

但是这样带来了一个风险,因为id是界面上客户输入的,所以如果没有进入校验,有人输入了一个aa' or '1'='1  把这个值代入到上面的sql语句里面,sql语句就变成了

1
SELECT ID,NAME FROM USER WHERE ID='aa' or '1'='1'

这样就能查到所有的数据了,也就是SQL注入

 

但是,如果用preparedstatement的话,就没有这个问题

1
String SQL = "SELECT ID,NAME FROM USER WHERE ID=?"

然后再将值set进去,如果值里面有引号等字符时,会自动的启用转义,不会破坏这个SQL语句的结果,也就不会造成SQL注入了

 大家都知道,Java中JDBC中,有个预处理功能,这个功能一大优势就是能提高执行速度尤其是多次操作数据库的情况,再一个优势就是预防SQL注入,严格的说,应该是预防绝大多数的SQL注入。

       用法就是如下边所示:

[java] view plain copy
 
  1. String sql="update cz_zj_directpayment dp"+  
  2.  "set dp.projectid = ? where dp.payid= ?";  
  3. try {  
  4.     PreparedStatement pset_f = conn.prepareStatement(sql);  
  5.     pset_f.setString(1,inds[j]);  
  6.     pset_f.setString(2,id);  
  7.     pset_f.executeUpdate(sql_update);  
  8. }catch(Exception e){  
  9.     //e.printStackTrace();  
  10.     logger.error(e.message());  
  11. }  


       那为什么它这样处理就能预防SQL注入提高安全性呢?其实是因为SQL语句在程序运行前已经进行了预编译,在程序运行时第一次操作数据库之前,SQL语句已经被数据库分析,编译和优化,对应的执行计划也会缓存下来并允许数据库已参数化的形式进行查询,当运行时动态地把参数传给PreprareStatement时,即使参数里有敏感字符如 or '1=1'也数据库会作为一个参数一个字段的属性值来处理而不会作为一个SQL指令,如此,就起到了SQL注入的作用了!

推荐阅读