执行sql语句为什么?用PreparedStatement要比Statement好用
PreparedStatement
public interface PreparedStatement extends Statement;可以看到PreparedStatement是Statement的子接口,我们在执行查询或者更新数据表数据的时候,拼写SQL语句是一个很费力并且容易出错的事情,PreparedStatement可以简化这样的一个过程.
PreParedStatement
1).why?我们为什么要使用它
使用Statement需要进行拼写SQl语句,辛苦并且容易出错,之前使用Statement的SQL语句的形式是这样的
String sql = "insert into examstudent" + " values("
+ student.getFlowId() + "," + student.getType() + ",'"
+ student.getIdCard() + "','" + student.getExamCard() + "','"
+ student.getStudentName() + "','" + student.getLocation()
+ "'," + student.getGrade() + ")";
使用PreparedStatement:是Statement的子接口,可以传入带占位符的SQL语句,提供了补充占位符变量的方法
PreparedStatement ps=conn.preparedStatement(sql);
可以看到将sql作为参数传入了,就不需要我们在费力拼写了。
2)变成了这样的形式
String sql="insert into examstudent values(?,?,?,?,?,?,?)";
可以调用PreparedStatement的setXxx(int index,Object val)设置占位符的值,其中index的值从1开始
执行SQl语句:excuteQuery()或者excuteUpdate()就可以完成查询或者数据的更新.【注意】:此时函数的参数位置不需要传入SQL语句,注意同使用Statement的update函数的差别
具体代码实现:

1 @Test
2 public void testPreparedStatement() {
3 Connection connection = null;
4 PreparedStatement preparedStatement = null;
5 try {
6 // 连接数据库
7 connection = JDBCTools.getConnection();
8 // 使用占位符的SQl语句
9 String sql = "insert into customers(name,email,birth)"
10 + "values(?,?,?)";
11 // 使用preparedStatement的setXxx方法设置每一个位置上的值
12 preparedStatement = connection.prepareStatement(sql);
13 // 设置name字段
14 preparedStatement.setString(1, "ATGUIGU");
15 // 设置email字段
16 preparedStatement.setString(2, "simale@163.com");
17 // 设置birth字段
18 preparedStatement.setDate(3,
19 new Date(new java.util.Date().getTime()));
20 // 执行更新操作
21 preparedStatement.executeUpdate();
22 } catch (Exception e) {
23 e.printStackTrace();
24 } finally {
25 // 释放资源
26 JDBCTools.release(null, preparedStatement, connection);
27 }
28 }

使用PreparedStatement执行SQl(更新操作:插入、删除、更新,但不包括select查询操作),JDBCTools中的通用函数update更改成下面的形式:这里使用了可变参数,而不是使用数组

 1     public static void update(String sql,Object ...args){
 2         /**
 3          * 执行SQL语句,使用PreparedStatement
 4          */
 5         Connection connection=null;
 6         PreparedStatement preparedStatement=null;
 7         try {
 8             connection=JDBCTools.getConnection();
 9             preparedStatement=connection.prepareStatement(sql);
10             for(int i=0;i<args.length;i++){
11                 preparedStatement.setObject(i+1, args[i]);
12             }
13             preparedStatement.executeUpdate();
14         } catch (Exception e) {
15             e.printStackTrace();
16         }finally{
17             JDBCTools.release(null, preparedStatement, connection);
18         }
19     }

使用PreparedStatement的好处:
1).提高代码的可读性和可维护性;
2).最大程度的提高性能:JDBC驱动的最佳化是基于使用的是什么功能. 选择PreparedStatement还是Statement取决于你要怎么使用它们. 对于只执行一次的SQL语句选择Statement是最好的. 相反, 如果SQL语句被多次执行选用PreparedStatement是最好的.PreparedStatement的第一次执行消耗是很高的. 它的性能体现在后面的重复执行(缓存的作用). 例如, 假设我使用Employee ID, 使用prepared的方式来执行一个针对Employee表的查询. JDBC驱动会发送一个网络请求到数据解析和优化这个查询. 而执行时会产生另一个网络请求. 在JDBC驱动中,减少网络通讯是最终的目的. 如果我的程序在运行期间只需要一次请求, 那么就使用Statement. 对于Statement, 同一个查询只会产生一次网络到数据库的通讯.当使用PreparedStatement池时, 如果一个查询很特殊, 并且不太会再次执行到, 那么可以使用Statement. 如果一个查询很少会被执行,但连接池中的Statement池可能被再次执行, 那么请使用PreparedStatement. 在不是Statement池的同样情况下, 请使用Statement.
3).可以防止SQL注入
SQL注入指的是通过构建特殊的输入作为参数传入Web应用程序,而这些输入大都是SQL语法里的一些组合,通过执行SQL语句进而执行攻击者所要的操作,其主要原因是程序没有细致地过滤用户输入的数据,致使非法数据侵入系统。
比如我们新建一个数据表users,表中有两个字段username和password;

我们在图形化界面SQLyog的sql语句的查询界面输入这样的查询语句:select * from users where username='a' or password='and password=' or '1'='1';
执行该语句,会得到我们表中的数据:

我们可以分析一下这条语句:where的后面,通过多个字段的组合作为查询过滤的条件。
字段一:username='a'
字段二:password='and password='
字段三:'1'='1'
因为用逻辑连接符OR来连接的三个字段,只要有一个为真就可以将查询工作完成.
下面我们看下具体的代码实现:

1 @Test
2 public void testSQLinjection() {
3 String username = "a' or password =";
4 String password = " or '1'='1";
5 String sql = "select * from users where username='" + username
6 + "' AND " + "password='" + password + "'";
7 System.out.println(sql);
8 Connection connection = null;
9 Statement statement = null;
10 ResultSet resultSet = null;
11 try {
12 connection = getConnection();
13 statement = connection.createStatement();
14 resultSet = statement.executeQuery(sql);
15 if (resultSet.next()) {
16 System.out.println("登陆成功");
17 } else {
18 System.out.println("不匹配");
19 }
20 } catch (Exception e) {
21 e.printStackTrace();
22 } finally {
23 JDBCTools.release(resultSet, statement, connection);
24 }
25 }

运行结果:
select * from users where username='a' or password =' AND password=' or '1'='1'
登陆成功
可以看到我们的SQl语句中都没有明确我们要查的字段的名,但是还是获取了查询的结果(SQL语句太能混了)
于是,我们用了PreparedStatement就可以解决SQL注入的问题。

1 @Test
2 public void testSQLinjection2() {
3 String username = "a' or password =";
4 String password = " or '1'='1";
5 String sql = "select * from users where username=?" + " and password=?";
6 System.out.println(sql);
7 Connection connection = null;
8 PreparedStatement preparedStatement = null;
9 ResultSet resultSet = null;
10 try {
11 connection = getConnection();
12 preparedStatement = connection.prepareStatement(sql);
13 preparedStatement.setString(1, username);
14 preparedStatement.setString(2, password);
15 resultSet = preparedStatement.executeQuery();
16 if (resultSet.next()) {
17 System.out.println("登陆成功");
18 } else {
19 System.out.println("不匹配");
20 }
21 } catch (Exception e) {
22 e.printStackTrace();
23 } finally {
24 JDBCTools.release(resultSet, preparedStatement, connection);
25 }
26 }

执行结果:
select * from users where username=? and password=?
不匹配
可以看到:再次使用伪装后的SQL语句已经不能获取我们数据表中的信息,我们这里在sql语句中使用了占位符。因此使用PreparedStatement可以结解决这里的SQL注入的问题。
<%@ page language="java" import="java.sql.*" pageEncoding="UTF-8"%>
<html>
<head>
<title>插入界面</title>
</head> <body>
<%
String driverName = "com.microsoft.sqlserver.jdbc.SQLServerDriver";
//加载JDBC驱动
String dbURL = "jdbc:sqlserver://localhost:1433; DatabaseName=JSPTest";
//连接服务器和数据库
String userName = "sa"; //默认用户名
String userPwd = "123456"; //密码
Class.forName(driverName);
Connection Conn = DriverManager.getConnection(dbURL, userName, userPwd);
System.out.println("Connection Successful!");
String sql="Insert into regist(ID,password) values(?,?)";
PreparedStatement pstmt=Conn.prepareStatement(sql);
pstmt.setString(1, "201504");
pstmt.setInt(2, 111);
int n=pstmt.executeUpdate();
if(n==1){%>数据插入成功!<br><%}
else{%>数据插入失败!<br><%}
%>
</body>
</html>
执行sql语句为什么?用PreparedStatement要比Statement好用的更多相关文章
- 使用预处理PreparedStatement执行Sql语句
		
/** * 使用预处理的方式执行Sql * @param sql Sql语句 * @param obj 变量值数组 * @return 查询结果 * @throws SQLException */ p ...
 - [疯狂Java]JDBC:PreparedStatement预编译执行SQL语句
		
1. SQL语句的执行过程——Statement直接执行的弊病: 1) SQL语句和编程语言一样,仅仅就会普通的文本字符串,首先数据库引擎无法识别这种文本字符串,而底层的CPU更不理解这些文本字符串( ...
 - JDBC进阶之PreparedStatement执行SQL语句(MySQL)
		
一.什么是PreparedStatement 参阅Java API文档,我们可以知道,PreparedStatement是Statement的子接口(如图所示),表示预编译的 SQ ...
 - Statement和PreparedStatement都是用来发送和执行SQL语句的
		
Statement和PreparedStatement都是用来发送和执行SQL语句的 DriverManager管理一组驱动程序
 - 三种执行SQL语句的的JAVA代码
		
问题描述: 连接数据库,执行SQL语句是必不可少的,下面给出了三种执行不通SQL语句的方法. 1.简单的Statement执行SQL语句.有SQL注入,一般不使用. public static voi ...
 - 在mybatis执行SQL语句之前进行拦击处理
		
转载自:http://blog.csdn.net/hfmbook/article/details/41985853 比较适用于在分页时候进行拦截.对分页的SQL语句通过封装处理,处理成不同的分页sql ...
 - JDBC详解系列(四)之建立Stament和执行SQL语句
		
建立Stament 在获得连接之后,我们就可以跟数据库进行交互了. 在JDBC中,我们发送SQL语句到数据库这些操作时通过Stament,Preparement,CallableStateme ...
 - JDBC中执行SQL语句的方式
		
一.执行DDL.DML语句 DDL.DML分别表示数据库定义语言.数据库操纵语言,操控这两种语言应该使用Statement对象的executeUpdate方法. 代码如下: public static ...
 - executeBatch()批量执行Sql语句
		
executeBatch()方法:用于成批地执行SQL语句,但不能执行返回值是ResultSet结果集的SQL语句,而是直接执行stmt.executeBatch(); addBatch():向批处理 ...
 
随机推荐
- hdoj4685
			
数据: /*999993 43 1 2 42 2 32 3 4*/ #include <iostream> #include <cstdio> #include <cma ...
 - php 添加环境变量
			
1.php添加环境变量主要为了能在 cmd和软件的客户端用php来运行 首先我们要做的第一步: 添加环境变量(记住php.exe的路径,然后再环境变量中编辑path 多个用逗号分隔开,保存重启电脑) ...
 - CRM SALES ORDER ADD EEWB SEARCH FIELD
			
1.找到对应的查询结构: T-code:GENIL_MODEL_BROWSER Component Set 输入'ONEORDER' 点击DISPLAY 展开Dynamic Query Object, ...
 - Weka中数据挖掘与机器学习系列之Weka系统安装(四)
			
能来看我这篇博客的朋友,想必大家都知道,Weka采用Java编写的,因此,具有Java“一次编译,到处运行”的特性.支持的操作系统有Windows x86.Windows x64.Mac OS X.L ...
 - learning makefile var
 - SSM 框架 整合<SpringMVC+Spring+MyBatis>
			
一 框架的搭建1.建立一个maven项目 2.建立五个module(entity,dao,service,action,web-view) 3.给予它们之间的依赖关系 dao-->entity ...
 - Mysql 截取日期的方法
			
//显示年月日 select date_format(date ,'%Y-%m-%d' ) from talbe_a //根据年月日分组 select date_format(date ,'%Y-%m ...
 - Backpack VI
			
Given an integer array nums with all positive numbers and no duplicates, find the number of possible ...
 - javascript 跨域请求详细分析(终极跨域解决办法)
			
自从我接触前端以来,接手的项目里面很大部分都是前后端分离的,后端只提供接口,前端根据后端接口渲染出实际页面.个人觉得这是一个挺好的模式,前后端各自负责各自的模块,分工明确,而且也给前端更大的发挥空间. ...
 - php源码学习——开篇
			
这个系列是对php源码的学习记录.由于本人水平有限,可能并不能写的非常清晰和深入,所以,可能只适合本人阅读:) 初次接触php源码,看到陌生的文件夹和大量的文件,可能会觉得茫然无措.php-inter ...