SQL注入标题
SQL注入标题说的是:用户输入的信息中含有SQL语句关键字,和步伐中的SQL语句举行字符串拼接,导致步伐中的SQL语句改变了原意。(SQL注入标题是一种系统安全标题)接下来我们来演示一下SQL注入标题。以用户登录为例。使用表:t_user业务形貌:系统启动后,给出登录页面,用户可以输入用户名和密码,用户名和密码全部正确,则登录成功,反之,则登录失败。分析一下要执行怎样的SQL语句?是不是如许的?
- select * from t_user where name = 用户输入的用户名 and password = 用户输入的密码;
复制代码 假如以上的SQL语句可以大概查询到结果,说明用户名和密码是正确的,则登录成功。假如查不到,说明是错误的,则登录失败。代码实现如下:
- package com.powernode.jdbc;
- import java.sql.*;
- import java.util.ResourceBundle;
- import java.util.Scanner;
- /**
- * 用户登录案例演示SQL注入问题
- */
- public class JDBCTest02 {
- public static void main(String[] args) {
- // 输出欢迎页面
- System.out.println("欢迎使用用户管理系统,请登录!");
- // 接收用户名和密码
- Scanner scanner = new Scanner(System.in);
- System.out.print("用户名:");
- String loginName = scanner.nextLine();
- System.out.print("密码:");
- String loginPwd = scanner.nextLine();
- // 读取属性配置文件,获取连接数据库的信息。
- ResourceBundle bundle = ResourceBundle.getBundle("com.powernode.jdbc.jdbc");
- String driver = bundle.getString("driver");
- String url = bundle.getString("url");
- String user = bundle.getString("user");
- String password = bundle.getString("password");
- // JDBC程序验证用户名和密码是否正确
- Connection conn = null;
- Statement stmt = null;
- ResultSet rs = null;
- try {
- // 1.注册驱动
- Class.forName(driver);
- // 2.获取连接
- conn = DriverManager.getConnection(url, user, password);
- // 3.获取数据库操作对象
- stmt = conn.createStatement();
- // 4.执行SQL语句
- String sql = "select realname from t_user where name = '"+loginName+"' and password = '"+loginPwd+"'";
- rs = stmt.executeQuery(sql);
- // 5.处理查询结果集
- if (rs.next()) { // 如果可以确定结果集中最多只有一条记录的话,可以使用if语句,不一定非要用while循环。
- String realname = rs.getString("realname");
- System.out.println("登录成功,欢迎您" + realname);
- } else {
- System.out.println("登录失败,用户名不存在或者密码错误。");
- }
- } catch (ClassNotFoundException | SQLException e) {
- throw new RuntimeException(e);
- } finally {
- // 6.释放资源
- if (rs != null) {
- try {
- rs.close();
- } catch (SQLException e) {
- throw new RuntimeException(e);
- }
- }
- if (stmt != null) {
- try {
- stmt.close();
- } catch (SQLException e) {
- throw new RuntimeException(e);
- }
- }
- if (conn != null) {
- try {
- conn.close();
- } catch (SQLException e) {
- throw new RuntimeException(e);
- }
- }
- }
- }
- }
复制代码- /**
- * ClassName: JDBCTest08
- * Description: 实现用户登录功能
- * 这个程序在实现了基本的登录功能的同时,演示了SQL注入的现象。
- * 导致SQL注入现象的根本原因是什么?
- * 第一:用户提供的信息中含有SQL语句的关键字了。
- * 第二:这些用户提供的信息中的关键字参与了SQL语句的编译。
- *
- * SQL注入是一种安全隐患。黑客通常使用SQL注入来攻击你的系统。
- * 怎么避免SQL注入现象呢?
- * java.sql.Statement 是存在SQL注入现象的,因为它的原理是:先拼接SQL语句字符串,然后再进行编译,所以它必然存在SQL注入的问题。
- * 为了避免SQL注入的发生,JDBC API中为Statement接口提供了一个子接口:java.sql.PreparedStatement,被称为预编译的数据库操作对象。
- * java.sql.PreparedStatement可以解决SQL注入问题。
- * 具体怎么解决的?PreparedStatement可以对SQL语句进行预先编译,然后给编译好的SQL语句占位符传值,通过这种方式来解决SQL注入问题。
- * 最本质的解决方法是:用户虽然提供了SQL语句关键字,但是这些关键字不再参与SQL语句的编译了。因此解决了SQL注入的问题。
- *
复制代码 假如用户名和密码正确的话,执行结果如下:
假如用户名不存在或者密码错误的话,执行结果如下:
接下来,见证古迹的时候,当我分别输入以下的用户名和密码时,系统被攻破了:
这种征象就叫做:SQL注入。为什么会发生以上的事儿呢?原因是:用户提供的信息中有SQL语句关键字,并且和底层的SQL字符串举行了拼接,变成了一个全新的SQL语句。例如:原来步伐想表达的是如许的SQL:
- select realname from t_user where name = 'sunwukong' and password = '123';
复制代码 结果被SQL注入之后,SQL语句就变成如许了:
- select realname from t_user where name = 'aaa' and password = 'bbb' or '1'='1';
复制代码 我们可以执行一下这条SQL,看看结果是什么?
把所有结果全部查到了,这是因为 '1'='1' 是恒成立的,并且使用的是 or 运算符,所以 or 前面的条件便是是没有的。如许就会把所有数据全部查到。而在步伐中的判断逻辑是只要结果集中有数据,则表示登录成功。所以以上的输入方式最终的结果就是登录成功。你设想一下,假如这个系统是一个高级别保密系统,只有登录成功的人才有权限,那么这个系统是不是极其伤害了。
解决SQL注入标题
导致SQL注入的根本原因是什么?只有找到真正的原因,标题才气得到解决。
最根本的原因是:Statement造成的。先举行SQL语句的字符串拼接,然后再举行SQL语句的编译
Statement执行原理是:先举行字符串的拼接,将拼接好的SQL语句发送给数据库服务器,数据库服务器举行SQL语句的编译,然后执行。因此用户提供的信息中假如含有SQL语句的关键字,那么这些关键字恰好参加了SQL语句的编译,所以导致原SQL语句被扭曲。
因此,JDBC为了解决这个标题,引入了一个新的接口:PreparedStatement,我们称为:预编译的数据库操作对象。PreparedStatement是Statement接口的子接口。它俩是继承关系。
PreparedStatement执行原理是:先对SQL语句举行预先的编译,然后再向SQL语句指定的位置传值,也就是说:用户提供的信息中纵然含有SQL语句的关键字,那么这个信息也只会被当做一个值转达给SQL语句,用户提供的信息不再参与SQL语句的编译了,如许就解决了SQL注入标题。
使用PreparedStatement解决SQL注入标题:
- package com.powernode.jdbc;
- import java.sql.*;
- import java.util.ResourceBundle;
- import java.util.Scanner;
- /**
- * PreparedStatement解决SQL注入问题
- */
- public class JDBCTest03 {
- public static void main(String[] args) {
- // 输出欢迎页面
- System.out.println("欢迎使用用户管理系统,请登录!");
- // 接收用户名和密码
- Scanner scanner = new Scanner(System.in);
- System.out.print("用户名:");
- String loginName = scanner.nextLine();
- System.out.print("密码:");
- String loginPwd = scanner.nextLine();
- // 读取属性配置文件,获取连接数据库的信息。
- ResourceBundle bundle = ResourceBundle.getBundle("com.powernode.jdbc.jdbc");
- String driver = bundle.getString("driver");
- String url = bundle.getString("url");
- String user = bundle.getString("user");
- String password = bundle.getString("password");
- // JDBC程序验证用户名和密码是否正确
- Connection conn = null;
- PreparedStatement pstmt = null;
- ResultSet rs = null;
- try {
- // 1.注册驱动
- Class.forName(driver);
- // 2.获取连接
- conn = DriverManager.getConnection(url, user, password);
- // 3.获取数据库操作对象(获取的是预编译的数据库操作对象)
- String sql = "select realname from t_user where name=? and password=?";
- pstmt = conn.prepareStatement(sql);
- pstmt.setString(1, loginName);
- pstmt.setString(2, loginPwd);
- // 4.执行SQL语句
- rs = pstmt.executeQuery();
- // 5.处理查询结果集
- if (rs.next()) {
- String realname = rs.getString("realname");
- System.out.println("登录成功,欢迎您" + realname);
- } else {
- System.out.println("登录失败,用户名不存在或者密码错误。");
- }
- } catch (ClassNotFoundException | SQLException e) {
- throw new RuntimeException(e);
- } finally {
- // 6.释放资源
- if (rs != null) {
- try {
- rs.close();
- } catch (SQLException e) {
- throw new RuntimeException(e);
- }
- }
- if (pstmt != null) {
- try {
- pstmt.close();
- } catch (SQLException e) {
- throw new RuntimeException(e);
- }
- }
- if (conn != null) {
- try {
- conn.close();
- } catch (SQLException e) {
- throw new RuntimeException(e);
- }
- }
- }
- }
- }
复制代码- /**
- * ClassName: JDBCTest09
- * Description: 使用PreparedStatement解决SQL注入问题。
- *
- * PreparedStatement原理:
- * 第一步:先对SQL语句进行预先编译
- * 第二步:给SQL语句中占位符传值
- *
- * 重点注意事项:
- * 在使用预编译的数据库操作对象PreparedStatement时,需要先编写SQL语句,然后再获取PreparedStatement对象。
- * 这里编写的SQL语句中,所有“值”的位置都要使用占位符来代替,占位符采用 ?
- * 每一个问号 ? 是一个值。是一个占位符。
- * 另外,特别要注意:这个占位符问号 ? 两边不能使用单引号或双引号括起来。
- *
- * 解决SQL注入的本质是:先将带有占位符的SQL语句进行预先编译,然后给占位符传值。
- * 即使用户提供的信息中含有SQL语句关键字,但是这些关键字不会参与SQL语句的编译,自然不会扭曲SQL语句的原意。
- *
- * Datetime: 2024/4/11 17:45
- * Author: 老杜@动力节点
- * Version: 1.0
- */
- public class JDBCTest09 {
- public static void main(String[] args) {
- Scanner scanner = new Scanner(System.in);
- // 初始化登录界面
- System.out.println("欢迎使用用户管理系统,请登录!");
- System.out.print("用户名:");
- String loginName = scanner.nextLine();
- System.out.print("密码:");
- String loginPwd = scanner.nextLine();
- // 连接数据库,验证用户名和密码是否正确。
- Connection conn = null;
- PreparedStatement ps = null;
- ResultSet rs = null;
- boolean loginSuccess = false;
- String realname = null;
- try {
- // 获取连接
- conn = DbUtils.getConnection();
- // 获取预编译的数据库操作对象
- String sql = "select * from t_user where name = ? and password = ?";
- ps = conn.prepareStatement(sql);
- // 给 ? 占位符 传值
- // 再次强调:在JDBC当中,所有的下标都是从1开始。不是从0开始。
- // 以下代码的含义是:给第1个占位符 ? 传值
- ps.setString(1, loginName);
- // 以下代码的含义是:给第2个占位符 ? 传值
- ps.setString(2, loginPwd);
- // 执行SQL语句
- rs = ps.executeQuery();
- // 处理结果集(结果集中有数据,表示登录成功,反之表示登录失败)
- if(rs.next()){
- // 登录成功
- loginSuccess = true;
- // 获取真实的名字
- realname = rs.getString("realname");
- }
- } catch (SQLException e) {
- throw new RuntimeException(e);
- } finally {
- // 释放资源
- DbUtils.close(conn, ps, rs);
- }
- System.out.println(loginSuccess ? "登录成功,欢迎" + realname : "登录失败,您的用户名不存在或者密码错误!");
- }
- }
复制代码 用户名和密码正确的话,执行结果如下:
data:image/s3,"s3://crabby-images/7a0e3/7a0e39b9dc0354451420e319399b4148788531e4" alt=""
用户名和密码错误的话,执行结果如下:
data:image/s3,"s3://crabby-images/73ba8/73ba8f01be2dd72851ec71700d908453c1f529d0" alt=""
实验SQL注入,看看还能不能?
data:image/s3,"s3://crabby-images/e5485/e5485126694f692185ec571bc0d94ee82dbd4a70" alt=""
通过测试得知,SQL注入标题已经解决了。根本原因是:bbb' or '1'='1 这个字符串中虽然含有SQL语句的关键字,但是只会被当做普通的值传到SQL语句中,并没有参与SQL语句的编译。
关于使用PreparedStatement要留意的是:
- 带有占位符 ? 的SQL语句我们称为:预处理SQL语句。
- 占位符 ? 不能使用单引号或双引号包裹。假如包裹,占位符则不再是占位符,是一个普通的问号字符。
- 在执行SQL语句前,必须给每一个占位符 ? 传值。
- 如何给占位符 ? 传值,通过以下的方法:
- pstmt.setXxx(第几个占位符, 传什么值)
- “第几个占位符”:从1开始。第1个占位符则是1,第2个占位符则是2,以此类推。
- “传什么值”:具体要看调用的什么方法?
- 假如调用pstmt.setString方法,则传的值必须是一个字符串。
- 假如调用pstmt.setInt方法,则传的值必须是一个整数。
- 以此类推......
PreparedStatement和Statement都是用于执行SQL语句的接口,它们的主要区别在于:
- PreparedStatement预编译SQL语句,Statement直接提交SQL语句;
- PreparedStatement执行速率更快,可以避免SQL注入攻击;(PreparedStatement对于同一条SQL语句来说,编译一次,执行N次。而Statement是每次都要举行编译的。因此PreparedStatement服从略微高一些。)
- PreparedStatement会做范例检查,是范例安全的;
PreparedStatement的使用
新增操作
需求:向 emp 表中插入如许一条记载:
empno:8888
ename:张三
job:贩卖员
mgr:7369
hiredate:2024-01-01
sal:1000.0
comm:500.0
deptno:10
- package com.powernode.jdbc;
- import java.sql.Connection;
- import java.sql.DriverManager;
- import java.sql.PreparedStatement;
- import java.sql.SQLException;
- import java.time.LocalDate;
- import java.util.ResourceBundle;
- public class JDBCTest04 {
- public static void main(String[] args) {
- ResourceBundle bundle = ResourceBundle.getBundle("com.powernode.jdbc.jdbc");
- String driver = bundle.getString("driver");
- String url = bundle.getString("url");
- String user = bundle.getString("user");
- String password = bundle.getString("password");
- Connection conn = null;
- PreparedStatement pstmt = null;
- try {
- // 1. 注册驱动
- Class.forName(driver);
- // 2. 获取连接
- conn = DriverManager.getConnection(url, user, password);
- // 3. 获取预编译的数据操作对象
- String sql = "insert into emp(empno,ename,sal,comm,job,mgr,hiredate,deptno) values(?,?,?,?,?,?,?,?)";
- // 预编译SQL语句
- pstmt = conn.prepareStatement(sql);
- // 给 ? 传值
- pstmt.setInt(1, 8888);
- pstmt.setString(2, "张三");
- pstmt.setDouble(3, 10000.0);
- pstmt.setDouble(4, 500.0);
- pstmt.setString(5, "销售员");
- pstmt.setInt(6, 7369);
- LocalDate localDate = LocalDate.parse("2024-01-01");
- pstmt.setDate(7, java.sql.Date.valueOf(localDate));
- pstmt.setInt(8, 10);
- // 4. 执行SQL语句
- int count = pstmt.executeUpdate();
- if (1 == count) {
- System.out.println("成功更新" + count + "条记录");
- }
- } catch (Exception e) {
- e.printStackTrace();
- } finally {
- // 6. 释放资源
- if (pstmt != null) {
- try {
- pstmt.close();
- } catch (SQLException e) {
- throw new RuntimeException(e);
- }
- }
- if (conn != null) {
- try {
- conn.close();
- } catch (SQLException e) {
- throw new RuntimeException(e);
- }
- }
- }
- }
- }
复制代码 重点学习内容:如何给占位符 ? 传值。执行结果如下:
data:image/s3,"s3://crabby-images/e379b/e379b4968e530750b3eac0265dcf05f547b84620" alt=""
修改操作
需求:将员工编号为8888的员工,姓名修改为李四,岗位修改为产品司理,月薪修改为5000.0,其他稳固。
- package com.powernode.jdbc;
- import java.sql.Connection;
- import java.sql.DriverManager;
- import java.sql.PreparedStatement;
- import java.sql.SQLException;
- import java.time.LocalDate;
- import java.util.ResourceBundle;
- public class JDBCTest05 {
- public static void main(String[] args) {
- ResourceBundle bundle = ResourceBundle.getBundle("com.powernode.jdbc.jdbc");
- String driver = bundle.getString("driver");
- String url = bundle.getString("url");
- String user = bundle.getString("user");
- String password = bundle.getString("password");
- Connection conn = null;
- PreparedStatement pstmt = null;
- try {
- // 1. 注册驱动
- Class.forName(driver);
- // 2. 获取连接
- conn = DriverManager.getConnection(url, user, password);
- // 3. 获取预编译的数据操作对象
- String sql = "update emp set ename = ?, job = ?, sal = ? where empno = ?";
- // 预编译SQL语句
- pstmt = conn.prepareStatement(sql);
- // 给 ? 传值
- pstmt.setString(1, "李四");
- pstmt.setString(2, "产品经理");
- pstmt.setDouble(3, 5000.0);
- pstmt.setInt(4, 8888);
- // 4. 执行SQL语句
- int count = pstmt.executeUpdate();
- if (1 == count) {
- System.out.println("成功更新" + count + "条记录");
- }
- } catch (Exception e) {
- e.printStackTrace();
- } finally {
- // 6. 释放资源
- if (pstmt != null) {
- try {
- pstmt.close();
- } catch (SQLException e) {
- throw new RuntimeException(e);
- }
- }
- if (conn != null) {
- try {
- conn.close();
- } catch (SQLException e) {
- throw new RuntimeException(e);
- }
- }
- }
- }
- }
复制代码 执行结果如下:
data:image/s3,"s3://crabby-images/44f63/44f6321db5b0d313e49be1944a0b020d5832e794" alt=""
删除操作
需求:将员工编号为8888的删除。
- package com.powernode.jdbc;
- import java.sql.Connection;
- import java.sql.DriverManager;
- import java.sql.PreparedStatement;
- import java.sql.SQLException;
- import java.util.ResourceBundle;
- public class JDBCTest06 {
- public static void main(String[] args) {
- ResourceBundle bundle = ResourceBundle.getBundle("com.powernode.jdbc.jdbc");
- String driver = bundle.getString("driver");
- String url = bundle.getString("url");
- String user = bundle.getString("user");
- String password = bundle.getString("password");
- Connection conn = null;
- PreparedStatement pstmt = null;
- try {
- // 1. 注册驱动
- Class.forName(driver);
- // 2. 获取连接
- conn = DriverManager.getConnection(url, user, password);
- // 3. 获取预编译的数据操作对象
- String sql = "delete from emp where empno = ?";
- // 预编译SQL语句
- pstmt = conn.prepareStatement(sql);
- // 给 ? 传值
- pstmt.setInt(1, 8888);
- // 4. 执行SQL语句
- int count = pstmt.executeUpdate();
- if (1 == count) {
- System.out.println("成功更新" + count + "条记录");
- }
- } catch (Exception e) {
- e.printStackTrace();
- } finally {
- // 6. 释放资源
- if (pstmt != null) {
- try {
- pstmt.close();
- } catch (SQLException e) {
- throw new RuntimeException(e);
- }
- }
- if (conn != null) {
- try {
- conn.close();
- } catch (SQLException e) {
- throw new RuntimeException(e);
- }
- }
- }
- }
- }
复制代码 执行结果如下:
data:image/s3,"s3://crabby-images/af871/af87156764f52259a744f1dcaccb8c0357eebe9e" alt=""
含糊查询
需求:查询员工名字中第二个字母是 O 的。
- package com.powernode.jdbc;
- import java.sql.*;
- import java.util.ResourceBundle;
- public class JDBCTest07 {
- public static void main(String[] args) {
- ResourceBundle bundle = ResourceBundle.getBundle("com.powernode.jdbc.jdbc");
- String driver = bundle.getString("driver");
- String url = bundle.getString("url");
- String user = bundle.getString("user");
- String password = bundle.getString("password");
- Connection conn = null;
- PreparedStatement pstmt = null;
- ResultSet rs = null;
- try {
- // 1. 注册驱动
- Class.forName(driver);
- // 2. 获取连接
- conn = DriverManager.getConnection(url, user, password);
- // 3. 获取预编译的数据操作对象
- String sql = "select ename from emp where ename like ?";
- // 预编译SQL语句
- pstmt = conn.prepareStatement(sql);
- // 给 ? 传值
- pstmt.setString(1, "_O%");
- // 4. 执行SQL语句
- rs = pstmt.executeQuery();
- // 5. 处理查询结果集
- while (rs.next()) {
- String ename = rs.getString("ename");
- System.out.println(ename);
- }
- } catch (Exception e) {
- e.printStackTrace();
- } finally {
- // 6. 释放资源
- if (rs != null) {
- try {
- rs.close();
- } catch (SQLException e) {
- throw new RuntimeException(e);
- }
- }
- if (pstmt != null) {
- try {
- pstmt.close();
- } catch (SQLException e) {
- throw new RuntimeException(e);
- }
- }
- if (conn != null) {
- try {
- conn.close();
- } catch (SQLException e) {
- throw new RuntimeException(e);
- }
- }
- }
- }
- }
复制代码 执行结果如下:
data:image/s3,"s3://crabby-images/fcaa9/fcaa959eccacbc5c10b1c828e79d0918ac8a7e08" alt=""
通过这个例子主要告诉大家,步伐不能如许写:
- String sql = "select ename from emp where ename like '_?%'";
- pstmt.setString(1, "O");
复制代码 由于占位符 ? 被单引号包裹,因此这个占位符是无效的。
分页查询
对于MySQL来说,通用的分页SQL语句:
假设每页显示3条记载:pageSize = 3
第1页:limit 0, 3
第2页:limit 3, 3
第3页:limit 6, 3
第pageNo页:limit (pageNo - 1)*pageSize, pageSize
需求:查询所有员工姓名,每页显示3条(pageSize),显示第2页(pageNo)。
- package com.powernode.jdbc;
- import java.sql.*;
- import java.util.ResourceBundle;
- public class JDBCTest08 {
- public static void main(String[] args) {
- ResourceBundle bundle = ResourceBundle.getBundle("com.powernode.jdbc.jdbc");
- String driver = bundle.getString("driver");
- String url = bundle.getString("url");
- String user = bundle.getString("user");
- String password = bundle.getString("password");
- Connection conn = null;
- PreparedStatement pstmt = null;
- ResultSet rs = null;
- // 每页显示记录条数
- int pageSize = 3;
- // 显示第几页
- int pageNo = 2;
- try {
- // 1. 注册驱动
- Class.forName(driver);
- // 2. 获取连接
- conn = DriverManager.getConnection(url, user, password);
- // 3. 获取预编译的数据操作对象
- String sql = "select ename from emp limit ?, ?";
- // 预编译SQL语句
- pstmt = conn.prepareStatement(sql);
- // 给 ? 传值
- pstmt.setInt(1, (pageNo - 1) * pageSize);
- pstmt.setInt(2, pageSize);
- // 4. 执行SQL语句
- rs = pstmt.executeQuery();
- // 5. 处理查询结果集
- while (rs.next()) {
- String ename = rs.getString("ename");
- System.out.println(ename);
- }
- } catch (Exception e) {
- e.printStackTrace();
- } finally {
- // 6. 释放资源
- if (rs != null) {
- try {
- rs.close();
- } catch (SQLException e) {
- throw new RuntimeException(e);
- }
- }
- if (pstmt != null) {
- try {
- pstmt.close();
- } catch (SQLException e) {
- throw new RuntimeException(e);
- }
- }
- if (conn != null) {
- try {
- conn.close();
- } catch (SQLException e) {
- throw new RuntimeException(e);
- }
- }
- }
- }
- }
复制代码 执行结果如下:
data:image/s3,"s3://crabby-images/4086c/4086cb70d533842fef31f201efc6eb259cb926ef" alt=""
blob数据的插入和读取
预备一张表:t_img,三个字段,一个id主键,一个图片名字name,一个img。建表语句如下:
- create table `t_img` (
- `id` bigint primary key auto_increment,
- `name` varchar(255),
- `img` blob
- ) engine=innodb;
复制代码 预备一张图片:
data:image/s3,"s3://crabby-images/da43e/da43e7e23d4a09ac9c11062cb4b4e3732e6d5125" alt=""
需求1:向t_img 表中插入一张图片。
- package com.powernode.jdbc;
- import java.io.FileInputStream;
- import java.io.IOException;
- import java.io.InputStream;
- import java.sql.Connection;
- import java.sql.DriverManager;
- import java.sql.PreparedStatement;
- import java.sql.SQLException;
- import java.util.ResourceBundle;
- public class JDBCTest09 {
- public static void main(String[] args) {
- ResourceBundle bundle = ResourceBundle.getBundle("com.powernode.jdbc.jdbc");
- String driver = bundle.getString("driver");
- String url = bundle.getString("url");
- String user = bundle.getString("user");
- String password = bundle.getString("password");
- Connection conn = null;
- PreparedStatement pstmt = null;
- InputStream in = null;
- try {
- // 1. 注册驱动
- Class.forName(driver);
- // 2. 获取连接
- conn = DriverManager.getConnection(url, user, password);
- // 3. 获取预编译的数据操作对象
- String sql = "insert into t_img(img) values(?)";
- pstmt = conn.prepareStatement(sql);
- // 获取文件输入流
- in = new FileInputStream("d:/dog.jpg");
- pstmt.setBlob(1, in);
- // 4. 执行SQL语句
- int count = pstmt.executeUpdate();
- System.out.println("插入了" + count + "条记录");
- } catch (Exception e) {
- e.printStackTrace();
- } finally {
- // 6. 释放资源
- if (in != null) {
- try {
- in.close();
- } catch (IOException e) {
- throw new RuntimeException(e);
- }
- }
- if (pstmt != null) {
- try {
- pstmt.close();
- } catch (SQLException e) {
- throw new RuntimeException(e);
- }
- }
- if (conn != null) {
- try {
- conn.close();
- } catch (SQLException e) {
- throw new RuntimeException(e);
- }
- }
- }
- }
- }
复制代码 执行结果如下:
data:image/s3,"s3://crabby-images/04da1/04da13c784470411d75705e4067daa67702566eb" alt=""
需求2:从t_img 表中读取一张图片。(从数据库中读取一张图片保存到本地。)
- package com.powernode.jdbc;
- import java.io.FileOutputStream;
- import java.io.InputStream;
- import java.io.OutputStream;
- import java.sql.*;
- import java.util.ResourceBundle;
- public class JDBCTest10 {
- public static void main(String[] args) {
- ResourceBundle bundle = ResourceBundle.getBundle("com.powernode.jdbc.jdbc");
- String driver = bundle.getString("driver");
- String url = bundle.getString("url");
- String user = bundle.getString("user");
- String password = bundle.getString("password");
- Connection conn = null;
- PreparedStatement pstmt = null;
- ResultSet rs = null;
- try {
- // 1. 注册驱动
- Class.forName(driver);
- // 2. 获取连接
- conn = DriverManager.getConnection(url, user, password);
- // 3. 获取预编译的数据操作对象
- String sql = "select img from t_img where id = ?";
- pstmt = conn.prepareStatement(sql);
- pstmt.setInt(1, 1);
- // 4. 执行SQL语句
- rs = pstmt.executeQuery();
- // 5. 处理查询结果集
- if (rs.next()) {
- // 获取二进制大对象
- Blob img = rs.getBlob("img");
- // 获取输入流
- InputStream binaryStream = img.getBinaryStream();
- // 创建输出流,该输出流负责写到本地
- OutputStream out = new FileOutputStream("d:/dog2.jpg");
- byte[] bytes = new byte[1024];
- int readCount = 0;
- while ((readCount = binaryStream.read(bytes)) != -1) {
- out.write(bytes, 0, readCount);
- }
- out.flush();
- binaryStream.close();
- out.close();
- }
- } catch (Exception e) {
- e.printStackTrace();
- } finally {
- // 6. 释放资源
- if (rs != null) {
- try {
- rs.close();
- } catch (SQLException e) {
- throw new RuntimeException(e);
- }
- }
- if (pstmt != null) {
- try {
- pstmt.close();
- } catch (SQLException e) {
- throw new RuntimeException(e);
- }
- }
- if (conn != null) {
- try {
- conn.close();
- } catch (SQLException e) {
- throw new RuntimeException(e);
- }
- }
- }
- }
- }
复制代码 执行完毕之后,查看一下图片大小是否和原图片相同,打开看看是否可以正常显示。
JDBC批处理操作
预备一张商品表:t_product建表语句如下:
- create table t_product(
- id bigint primary key,
- name varchar(255)
- );
复制代码 不使用批处理
不使用批处理,向 t_product 表中插入一万条商品信息,并记载耗时!
- package com.powernode.jdbc;
- import java.sql.Connection;
- import java.sql.DriverManager;
- import java.sql.PreparedStatement;
- import java.sql.SQLException;
- import java.util.ResourceBundle;
- public class NoBatchTest {
- public static void main(String[] args) {
- ResourceBundle bundle = ResourceBundle.getBundle("com.powernode.jdbc.jdbc");
- String driver = bundle.getString("driver");
- String url = bundle.getString("url");
- String user = bundle.getString("user");
- String password = bundle.getString("password");
- long begin = System.currentTimeMillis();
- Connection conn = null;
- PreparedStatement pstmt = null;
- try {
- // 1. 注册驱动
- Class.forName(driver);
- // 2. 获取连接
- conn = DriverManager.getConnection(url, user, password);
- // 3. 获取预编译的数据操作对象
- String sql = "insert into t_product(id, name) values (?, ?)";
- pstmt = conn.prepareStatement(sql);
- int count = 0;
- for (int i = 1; i <= 10000; i++) {
- pstmt.setInt(1, i);
- pstmt.setString(2, "product" + i);
- // 4. 执行SQL语句
- count += pstmt.executeUpdate();
- }
- System.out.println("插入了" + count + "条记录");
- } catch (Exception e) {
- e.printStackTrace();
- } finally {
- // 6. 释放资源
- if (pstmt != null) {
- try {
- pstmt.close();
- } catch (SQLException e) {
- throw new RuntimeException(e);
- }
- }
- if (conn != null) {
- try {
- conn.close();
- } catch (SQLException e) {
- throw new RuntimeException(e);
- }
- }
- }
- long end = System.currentTimeMillis();
- System.out.println("总耗时" + (end - begin) + "毫秒");
- }
- }
复制代码 执行结果如下:
data:image/s3,"s3://crabby-images/1a404/1a4049ae20ac775511dbc138f86c8b0f12a0aacf" alt=""
使用批处理
使用批处理,向 t_product 表中插入一万条商品信息,并记载耗时!留意:启用批处理需要在URL背面添加这个的参数:rewriteBatchedStatements=true
data:image/s3,"s3://crabby-images/54139/54139b832479bcde617da2cd483264545c0be66f" alt=""
- package com.powernode.jdbc;
- import java.sql.Connection;
- import java.sql.DriverManager;
- import java.sql.PreparedStatement;
- import java.sql.SQLException;
- import java.util.ResourceBundle;
- public class BatchTest {
- public static void main(String[] args) {
- ResourceBundle bundle = ResourceBundle.getBundle("com.powernode.jdbc.jdbc");
- String driver = bundle.getString("driver");
- String url = bundle.getString("url");
- String user = bundle.getString("user");
- String password = bundle.getString("password");
- long begin = System.currentTimeMillis();
- Connection conn = null;
- PreparedStatement pstmt = null;
- try {
- // 1. 注册驱动
- Class.forName(driver);
- // 2. 获取连接
- conn = DriverManager.getConnection(url, user, password);
- // 3. 获取预编译的数据操作对象
- String sql = "insert into t_product(id, name) values (?, ?)";
- pstmt = conn.prepareStatement(sql);
- int count = 0;
- for (int i = 1; i <= 10000; i++) {
- pstmt.setInt(1, i);
- pstmt.setString(2, "product" + i);
- pstmt.addBatch();
- }
- // 循环结束之后,再次执行批处理,防止数据丢失。
- count += pstmt.executeBatch().length;
- System.out.println("插入了" + count + "条记录");
- } catch (Exception e) {
- e.printStackTrace();
- } finally {
- // 6. 释放资源
- if (pstmt != null) {
- try {
- pstmt.close();
- } catch (SQLException e) {
- throw new RuntimeException(e);
- }
- }
- if (conn != null) {
- try {
- conn.close();
- } catch (SQLException e) {
- throw new RuntimeException(e);
- }
- }
- }
- long end = System.currentTimeMillis();
- System.out.println("总耗时" + (end - begin) + "毫秒");
- }
- }
复制代码- public class JDBCTest18 {
- public static void main(String[] args) {
- long begin = System.currentTimeMillis();
- Connection conn = null;
- PreparedStatement ps = null;
- try {
- conn = DbUtils.getConnection();
- String sql = "insert into t_batch(id,name) values(?,?)";
- ps = conn.prepareStatement(sql);
- int count = 0;
- for (int i = 1; i <= 10000; i++) {
- ps.setLong(1, i);
- ps.setString(2, "batch" + i);
- // 打包
- ps.addBatch();
- // 如果打包够500个,则执行一次(则磁盘IO一次)
- if(i % 500 == 0){
- count += ps.executeBatch().length;
- }
- }
- // 循环结束之后,再次执行批处理,防止数据丢失。
- count += ps.executeBatch().length;
- System.out.println("插入了" + count + "条记录");
- } catch (SQLException e) {
- throw new RuntimeException(e);
- } finally {
- DbUtils.close(conn, ps, null);
- }
- long end = System.currentTimeMillis();
- System.out.println("总耗时" + (end - begin) + "毫秒"); // 总耗时1652毫秒
- }
- }
复制代码 执行结果如下:
在举行大数据量插入时,批处理为什么可以进步步伐的执行服从?
- 淘汰了网络通信次数:JDBC 批处理会将多个 SQL 语句一次性发送给服务器,淘汰了客户端和服务器之间的通信次数,从而进步了数据写入的速率,特别是对于远程服务器而言,优化结果更为明显。
- 淘汰了数据库操作次数:JDBC 批处理会将多个 SQL 语句归并成一条 SQL 语句举行执行,从而淘汰了数据库操作的次数,减轻了数据库的负担,大大进步了数据写入的速率。
DbUtils工具类的封装
JDBC编程六步中,许多代码是重复出现的,可以为这些代码封装一个工具类。让JDBC代码变的更简洁。
- package com.powernode.jdbc;
- import java.sql.*;
- import java.util.ResourceBundle;
- /**
- * ClassName: DbUtils
- * Description: JDBC工具类
- * Datetime: 2024/4/10 22:29
- * Author: 老杜@动力节点
- * Version: 1.0
- */
- public class DbUtils {
- private static String url;
- private static String user;
- private static String password;
- static {
- // 读取属性资源文件
- ResourceBundle bundle = ResourceBundle.getBundle("com.powernode.jdbc.jdbc");
- String driver = bundle.getString("driver");
- url = bundle.getString("url");
- user = bundle.getString("user");
- password = bundle.getString("password");
- // 注册驱动
- try {
- Class.forName(driver);
- } catch (ClassNotFoundException e) {
- throw new RuntimeException(e);
- }
- }
- /**
- * 获取数据库连接
- * @return
- * @throws SQLException
- */
- public static Connection getConnection() throws SQLException {
- Connection conn = DriverManager.getConnection(url, user, password);
- return conn;
- }
- /**
- * 释放资源
- * @param conn 连接对象
- * @param stmt 数据库操作对象
- * @param rs 结果集对象
- */
- public static void close(Connection conn, Statement stmt, ResultSet rs){
- if (rs != null) {
- try {
- rs.close();
- } catch (SQLException e) {
- throw new RuntimeException(e);
- }
- }
- if (stmt != null) {
- try {
- stmt.close();
- } catch (SQLException e) {
- throw new RuntimeException(e);
- }
- }
- if (conn != null) {
- try {
- conn.close();
- } catch (SQLException e) {
- throw new RuntimeException(e);
- }
- }
- }
- }
复制代码 免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。 |