java中PreparedStatement和Statement详细讲解

java中PreparedStatement和Statement详细讲解大家都知道PreparedStatement对象可以防止sql注入,而Statement不能防止sql注入,那么大家知道为什么PreparedStatement对象可以防止sql注入…

大家好,又见面了,我是你们的朋友全栈君。

java中PreparedStatement和Statement详细讲解

大家都知道PreparedStatement对象可以防止sql注入,而Statement不能防止sql注入,那么大家知道为什么PreparedStatement对象可以防止sql注入,接下来看我的案例大家就会明白了!

我用的是mysql数据库,以admin表为例子,如下图:

最后面有具体的java代码和sql代码案例

java中PreparedStatement和Statement详细讲解

java中PreparedStatement和Statement详细讲解

java中PreparedStatement和Statement详细讲解

java中PreparedStatement和Statement详细讲解

最终执行的sql语句打印出来是SELECT * FROM admin WHERE username = ‘韦小宝’ AND password = ‘222\’ OR \’8\’=\’8′

从以上截图就能看出来,由此可见,prepareStatement对象防止sql注入的方式是把用户非法输入的单引号用\反斜杠做了转义,从而达到了防止sql注入的目的

Statement对象就没那么好心了,它才不会把用户非法输入的单引号用\反斜杠做转义呢!

PreparedStatement可以有效防止sql注入,所以生产环境上一定要使用PreparedStatement,而不能使用Statement

当然啦,你可以仔细研究下PreparedStatement对象是如何防止sql注入的,我自己把最终执行的sql语句打印出来了,看到打印出来的sql语句就明白了,原来是mysql数据库产商,在实现PreparedStatement接口的实现类中的setString(int parameterIndex, String x)函数中做了一些处理,把单引号做了转义(只要用户输入的字符串中有单引号,那mysql数据库产商的setString()这个函数,就会把单引号做转义)

大家有兴趣可以去网上,下载一份mysql数据库的驱动程序的源代码,看看mysql数据库产商的驱动程序的源代码,去源代码中找到setString(int parameterIndex, String x)函数,看看该函数中是怎么写的,我没有下载mysql数据库产商的驱动程序的源代码,而是把mysql数据库的驱动程序jar包解压了,找到了PreparedStatement.class文件,利用反编译工具,反编译了一下,如下:

java中PreparedStatement和Statement详细讲解java中PreparedStatement和Statement详细讲解

java中PreparedStatement和Statement详细讲解

java中PreparedStatement和Statement详细讲解

java中PreparedStatement和Statement详细讲解

java中PreparedStatement和Statement详细讲解

java中PreparedStatement和Statement详细讲解

java中PreparedStatement和Statement详细讲解

java中PreparedStatement和Statement详细讲解

java中PreparedStatement和Statement详细讲解 

java中PreparedStatement和Statement详细讲解 

java中PreparedStatement和Statement详细讲解 这下大家应该知道PreparedStatement是如何防止sql注入的了吧

像222′ OR ‘8’=’8这样的sql注入还算温柔了,有些更可恶的用户,他们输入的非法的值是delete from tableName或truncate table tableName 这是十分危险的,更有甚者传入drop table tableName;有些数据库是不会让你成功的,但也有很多数据库就可以使这些语句执行,所以生产环境上一定要使用PreparedStatement,而不能使用Statement

下面再举几个例子,看截图

java中PreparedStatement和Statement详细讲解

最终打印SELECT * FROM admin WHERE username = ‘韦小宝’ AND password = ‘\’; DROP TABLE tableName;#’

java中PreparedStatement和Statement详细讲解

最终打印SELECT * FROM admin WHERE username = ‘韦小宝’ AND password = ‘\’; delete from tableName;#’

java中PreparedStatement和Statement详细讲解

最终打印SELECT * FROM admin WHERE username = ‘韦小宝’ AND password = ‘\’; truncate table tableName;#’

java中PreparedStatement和Statement详细讲解

下面是java代码和sql语句,供大家参考,主要是为了测试PreparedStatement对象,所以java代码写的比较粗略,大家凑合着看吧!

package com.test;

import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.sql.Statement;

/*
 * 研究PreparedStatement是如何防止sql注入的,我分析了一下,原来是mysql数据库产商,在实
 * 现PreparedStatement接口的实现类中的setString(int parameterIndex, String x)函
 * 数中做了一些处理,把单引号做了转义(只要用户输入的字符串中有单引号,那mysql数据库产商的setString()这个函
 * 数,就会把单引号做转义)
 */
public class TestConnMySql2 {

	public static void main(String[] args) {
		String connStr = "jdbc:mysql://localhost:3306/girls";
//		String sql = "select * from admin";
		String sql = "SELECT * FROM admin WHERE username = ? AND password = ?";
		try {
			Class.forName("com.mysql.jdbc.Driver");
			Connection connection = DriverManager.getConnection(connStr, "root", "root");
			System.out.println("数据库连接=" + connection);
			//Statement无法防止sql注入
//			Statement stmt = connection.createStatement();
	//PreparedStatement可以有效防止sql注入,所以生产环境上一定要使用PreparedStatement,而不能使用Statement
			PreparedStatement prepareStatement = connection.prepareStatement(sql);
			prepareStatement.setString(1, "韦小宝");
			//模拟用户输入正常的值
//			prepareStatement.setString(2, "222");
			//测试sql注入(模拟用户输入非法的值)
			prepareStatement.setString(2, "222' OR '8'='8");
			/*
			 *上面那种的sql注入还算温柔了,有些更可恶的用户,他们输入的非
			 *法的值是delete from tableName或truncate table tableName 这是十分危险的,
			 * 更有甚者传入drop table tableName;有些数据库是不会让你成功的,但也有很多数
			 * 据库就可以使这些语句执行,所以生产环境上一定要使用PreparedStatement,而不能使用Statement
			 */
			//测试sql注入(模拟用户输入非法的值)在mysql中#井号表示单行注释(这是mysql中的基础知识,我就不赘述了)
//			prepareStatement.setString(2, "'; DROP TABLE tableName;#");
			//测试sql注入(模拟用户输入非法的值)
//			prepareStatement.setString(2, "'; delete from tableName;#");
			//测试sql注入(模拟用户输入非法的值)
//			prepareStatement.setString(2, "'; truncate table tableName;#");
			
			ResultSet rs = prepareStatement.executeQuery();
			System.out.println("sql=" + prepareStatement.toString());
			int col = rs.getMetaData().getColumnCount();
			System.out.println("============================");
			while (rs.next()) {
				for (int i = 1; i <= col; i++) {
					System.out.print(rs.getString(i) + "\t");
					if ((i == 2) && (rs.getString(i).length() < 8)) {
						System.out.print("\t");
					}
				}
				System.out.println("");
			}
			System.out.println("============================");
			rs.close();
			prepareStatement.close();
			connection.close();
		} catch (ClassNotFoundException | SQLException e) {
			e.printStackTrace();
		}

	}

}

#用户输入正常合法的值
SELECT * FROM admin WHERE username = '韦小宝' AND `password` = '222';
#用户输入正常合法的值
SELECT * FROM admin WHERE username = '韦小宝' AND PASSWORD = '222';

#sql注入(用户输入非法的值)使用Statement对象,无法防止sql注入(会查询出表的所有数据)
SELECT * FROM admin WHERE username = '韦小宝' AND PASSWORD = '222' OR '8'='8'
#sql注入(用户输入非法的值)使用PreparedStatement对象,可以有效防止sql注入
SELECT * FROM admin WHERE username = '韦小宝' AND PASSWORD = '222\' OR \'8\'=\'8'

#sql注入(用户输入非法的值)使用Statement对象,无法防止sql注入(DROP操作很危险)
SELECT * FROM admin WHERE username = '韦小宝' AND PASSWORD = ''; DROP TABLE tableName;#'
#sql注入(用户输入非法的值)使用PreparedStatement对象,可以有效防止sql注入
SELECT * FROM admin WHERE username = '韦小宝' AND PASSWORD = '\'; DROP TABLE tableName;#'

#sql注入(用户输入非法的值)使用Statement对象,无法防止sql注入(TRUNCATE操作很危险)
SELECT * FROM admin WHERE username = '韦小宝' AND PASSWORD = ''; TRUNCATE TABLE tableName;#'
#sql注入(用户输入非法的值)使用PreparedStatement对象,可以有效防止sql注入
SELECT * FROM admin WHERE username = '韦小宝' AND PASSWORD = '\'; truncate table tableName;#'

#sql注入(用户输入非法的值)使用Statement对象,无法防止sql注入(DELETE操作很危险)
SELECT * FROM admin WHERE username = '韦小宝' AND PASSWORD = ''; DELETE FROM tableName;#'
#sql注入(用户输入非法的值)使用PreparedStatement对象,可以有效防止sql注入
SELECT * FROM admin WHERE username = '韦小宝' AND PASSWORD = '\'; delete from tableName;#'

#所以生产环境上一定要使用PreparedStatement,而不能使用Statement

/*
顺便复习一下mysql中的3种注释,我是多行注释
*/

#我是单行注释

-- 我也是单行注释(注意:-- 这种注释,后面必须要加一个空格,否则语法报错)


版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/130163.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • linux之文件操作

    1.文件操作思维导图2.linux系统目录结构及简单说明linux目录图:root启动Linux时使用的一些核心文件。如操作系统内核、引导程序Grub等。home存储普通用户的个人文件

    2021年12月28日
    45
  • kubernetes可以实现容器集群的哪些功能_hadoop高可用集群搭建

    kubernetes可以实现容器集群的哪些功能_hadoop高可用集群搭建二进制方式部署Kubernetes高可用集群文章目录二进制方式部署Kubernetes高可用集群1.环境准备1.1.Kubernetes高可用集群部署方式1.2.Kubernetes集群弃用docker容器1.3.Kubernetes集群所需的证书1.4.环境准备1.5.安装cfssl证书生成工具2.操作系统初始化配置3.部署Etcd集群3.1.使用cfssl证书工具生成etcd证书3.2.部署etcd集群4.部署Docker服务4.1.安装docker4.2.为docker创建systemctl启动脚本

    2025年8月28日
    6
  • Spring常用的三种注入方式「建议收藏」

    Spring常用的三种注入方式「建议收藏」Spring通过DI(依赖注入)实现IOC(控制反转),常用的注入方式主要有三种:构造方法注入,setter注入,基于注解的注入。构造方法注入先简单看一下测试项目的结构,用maven构建的,四个包:entity:存储实体,里面只有一个User类dao:数据访问,一个接口,两个实现类service:服务层,一个接口,一个实现类,实现类依赖于IUserDaotest:测试包在spring的配置文

    2025年8月21日
    2
  • DDD理论学习系列(13)– 模块

    DDD理论学习系列(13)– 模块"DDD理论学习系列——案例及目录"1.引言Module,即模块,是指提供特定功能的相对独立的单元。提到模块,你肯定就会想到模块化设计思想,也就是功能的分解和组合。对于简单问

    2022年7月3日
    24
  • 在宝塔上配置阿里SSL证书流程[通俗易懂]

    在宝塔上配置阿里SSL证书流程[通俗易懂]1.在阿里申请SSL证书2.下载申请好的ssl证书3.在宝塔上找到ssl4.复制证书安装顺序复制用文本打开.key文件复制里面的内容到左侧秘钥(key)用文本打开_public.crt文件复制里面的内容到右侧证书(pem)格式用文本打开_chain.crt文件复制里面的内容到右侧证书(pem)格式5.部署部署成功…

    2022年10月4日
    5
  • CBOW最强理解_创造之最强C位

    CBOW最强理解_创造之最强C位翻译自:https://iksinc.online/tag/continuous-bag-of-words-cbow/向量空间模型在信息检索中是众所周知的,其中每个文档被表示为向量。矢量分量表示文档中每个单词的权重或重要性。使用余弦相似性度量计算两个文档之间的相似性。尽管对单词使用矢量表示的想法也已经存在了一段时间,但是对于嵌入单词的技术,将单词映射到向量的技术,最近一直在飙升。其中一个驱…

    2025年9月29日
    2

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号