2022.4.23 JDBC API详解
JDBC API详解
DriverManager
DriverManager(驱动管理类)作用:
-
注册驱动
registerDriver方法是用于注册驱动的,但是我们之前做的入门案例并不是这样写的。而是如下实现
1 Class.forName("com.mysql.jdbc.Driver");
我们查询MySQL提供的Driver类,看它是如何实现的,源码如下:
在该类中的静态代码块中已经执行了
DriverManager
对象的registerDriver()
方法进行驱动的注册了,那么我们只需要加载Driver
类,该静态代码块就会执行。而Class.forName("com.mysql.jdbc.Driver");
就可以加载Driver
类。
-
-
MySQL 5之后的驱动包,可以省略注册驱动的步骤
-
自动加载jar包中META-INF/services/java.sql.Driver文件中的驱动类
-
-
获取数据库连接
参数说明:
-
url : 连接路径
语法:jdbc:mysql://ip地址(域名):端口号/数据库名称?参数键值对1&参数键值对2…
示例:jdbc:mysql://127.0.0.1:3306/db1
细节:
-
如果连接的是本机mysql服务器,并且mysql服务默认端口是3306,则url可以简写为:jdbc:mysql:///数据库名称?参数键值对
-
配置 useSSL=false 参数,禁用安全连接方式,解决警告提示
-
-
user :用户名
-
-
Connection
Connection(数据库连接对象)作用:
-
获取执行 SQL 的对象
-
管理事务
获取执行对象
-
普通执行SQL对象
1 Statement createStatement()
入门案例中就是通过该方法获取的执行对象。
-
预编译SQL的执行SQL对象:防止SQL注入
1 PreparedStatement prepareStatement(sql)
通过这种方式获取的
PreparedStatement
SQL语句执行对象是我们一会重点要进行讲解的,它可以防止SQL注入。 -
执行存储过程的对象
1 CallableStatement prepareCall(sql)
通过这种方式获取的
CallableStatement
执行对象是用来执行存储过程的,而存储过程在MySQL中不常用,所以这个我们将不进行讲解。
事务管理
先回顾一下MySQL事务管理的操作:
-
开启事务 : BEGIN; 或者 START TRANSACTION;
-
提交事务 : COMMIT;
-
回滚事务 : ROLLBACK;
MySQL默认是自动提交事务
接下来学习JDBC事务管理的方法。
Connection几口中定义了3个对应的方法:
-
开启事务
参与autoCommit 表示是否自动提交事务,true表示自动提交事务,false表示手动提交事务。而开启事务需要将该参数设为为false。
-
提交事务
-
回滚事务
具体代码实现如下:
1 /** 2 * JDBC API 详解:Connection 3 */ 4 public class JDBCDemo3_Connection { 5 6 public static void main(String[] args) throws Exception { 7 //1. 注册驱动 8 //Class.forName("com.mysql.jdbc.Driver"); 9 //2. 获取连接:如果连接的是本机mysql并且端口是默认的 3306 可以简化书写 10 String url = "jdbc:mysql:///db1?useSSL=false"; 11 String username = "root"; 12 String password = "1234"; 13 Connection conn = DriverManager.getConnection(url, username, password); 14 //3. 定义sql 15 String sql1 = "update account set money = 3000 where id = 1"; 16 String sql2 = "update account set money = 3000 where id = 2"; 17 //4. 获取执行sql的对象 Statement 18 Statement stmt = conn.createStatement(); 19 20 try { 21 // ============开启事务========== 22 conn.setAutoCommit(false); 23 //5. 执行sql 24 int count1 = stmt.executeUpdate(sql1);//受影响的行数 25 //6. 处理结果 26 System.out.println(count1); 27 int i = 3/0;//出现错误 会回滚 28 //5. 执行sql 29 int count2 = stmt.executeUpdate(sql2);//受影响的行数 30 //6. 处理结果 31 System.out.println(count2); 32 33 // ============提交事务========== 34 //程序运行到此处,说明没有出现任何问题,则需求提交事务 35 conn.commit(); 36 } catch (Exception e) { 37 // ============回滚事务========== 38 //程序在出现异常时会执行到这个地方,此时就需要回滚事务 39 conn.rollback(); 40 e.printStackTrace(); 41 } 42 43 //7. 释放资源 44 stmt.close(); 45 conn.close(); 46 } 47 }
Statement
概述
Jdbc中的statement对象用于向数据库发送SQL语句,想完成对数据库的增删改查,只需要通过这个对象向数据库发送增删改查语句即可。
Statement对象的作用就是用来执行SQL语句。而针对不同类型的SQL语句使用的方法也不一样。
-
执行DDL、DML语句
-
执行DQL语句
该方法涉及到了
ResultSet
对象,而这个对象我们还没有学习,一会再重点讲解。
代码实现
-
执行DML语句
1 /** 2 * 执行DML语句 3 * @throws Exception 4 */ 5 @Test 6 public void testDML() throws Exception { 7 //1. 注册驱动 8 //Class.forName("com.mysql.jdbc.Driver"); 9 //2. 获取连接:如果连接的是本机mysql并且端口是默认的 3306 可以简化书写 10 String url = "jdbc:mysql:///db1?useSSL=false"; 11 String username = "root"; 12 String password = "1234"; 13 Connection conn = DriverManager.getConnection(url, username, password); 14 //3. 定义sql 15 String sql = "update account set money = 3000 where id = 1"; 16 //4. 获取执行sql的对象 Statement 17 Statement stmt = conn.createStatement(); 18 //5. 执行sql 19 int count = stmt.executeUpdate(sql);//执行完DML语句,受影响的行数 20 //6. 处理结果 21 //System.out.println(count); 22 if(count > 0){ 23 System.out.println("修改成功~"); 24 }else{ 25 System.out.println("修改失败~"); 26 } 27 //7. 释放资源 28 stmt.close(); 29 conn.close(); 30 }
-
执行DDL语句
1 /** 2 * 执行DDL语句 3 * @throws Exception 4 */ 5 @Test 6 public void testDDL() throws Exception { 7 //1. 注册驱动 8 //Class.forName("com.mysql.jdbc.Driver"); 9 //2. 获取连接:如果连接的是本机mysql并且端口是默认的 3306 可以简化书写 10 String url = "jdbc:mysql:///db1?useSSL=false"; 11 String username = "root"; 12 String password = "1234"; 13 Connection conn = DriverManager.getConnection(url, username, password); 14 //3. 定义sql 15 String sql = "drop database db2"; 16 //4. 获取执行sql的对象 Statement 17 Statement stmt = conn.createStatement(); 18 //5. 执行sql 19 int count = stmt.executeUpdate(sql);//执行完DDL语句,可能是0,不能用count去判断成功与否 20 //6. 处理结果 21 System.out.println(count); 22 23 //7. 释放资源 24 stmt.close(); 25 conn.close(); 26 }
注意:以后开发很少使用java代码操作DDL语句,执行完DDL语句,可能是0,不能用count去判断成功与否
ResultSet
概述
ResultSet(结果集对象)作用:封装了SQL查询语句的结果。
而执行了DQL语句后就会返回该对象,对应执行DQL语句的方法如下:
ResultSet executeQuery(sql):执行DQL 语句,返回 ResultSet 对象
那么我们就需要从 ResultSet
对象中获取我们想要的数据。ResultSet
对象提供了操作查询结果数据的方法,如下:
boolean next()
-
将光标从当前位置向前移动一行
-
判断当前行是否为有效行
方法返回值说明:
-
true : 有效行,当前行有数据
-
false : 无效行,当前行没有数据
xxx getXxx(参数):获取数据
-
xxx : 数据类型;如: int getInt(参数) ;String getString(参数)
-
参数
-
int类型的参数:列的编号,从1开始
-
String类型的参数: 列的名称
-
如下图为执行SQL语句后的结果
一开始光标指定于第一行前,如图所示红色箭头指向于表头行。当我们调用了 next()
方法后,光标就下移到第一行数据,并且方法返回true,此时就可以通过 getInt("id")
获取当前行id字段的值,也可以通过 getString("name")
获取当前行name字段的值。如果想获取下一行的数据,继续调用 next()
方法,以此类推。
代码实现
1 /** 2 * 执行DQL 3 * @throws Exception 4 */ 5 @Test 6 public void testResultSet() throws Exception { 7 //1. 注册驱动 8 //Class.forName("com.mysql.jdbc.Driver"); 9 //2. 获取连接:如果连接的是本机mysql并且端口是默认的 3306 可以简化书写 10 String url = "jdbc:mysql:///db1?useSSL=false"; 11 String username = "root"; 12 String password = "1234"; 13 Connection conn = DriverManager.getConnection(url, username, password); 14 //3. 定义sql 15 String sql = "select * from account"; 16 //4. 获取statement对象 17 Statement stmt = conn.createStatement(); 18 //5. 执行sql 19 ResultSet rs = stmt.executeQuery(sql); 20 //6. 处理结果, 遍历rs中的所有数据 21 /* // 6.1 光标向下移动一行,并且判断当前行是否有数据 22 while (rs.next()){ 23 //6.2 获取数据 getXxx() int类型参数(第几列从1开始) 24 int id = rs.getInt(1); 25 String name = rs.getString(2); 26 double money = rs.getDouble(3); 27 28 System.out.println(id); 29 System.out.println(name); 30 System.out.println(money); 31 32 System.out.println("--------------"); 33 34 }*/ 35 // 6.1 光标向下移动一行,并且判断当前行是否有数据 36 while (rs.next()){ 37 //6.2 获取数据 getXxx() String类型参数(数据库列名) 38 int id = rs.getInt("id"); 39 String name = rs.getString("name"); 40 double money = rs.getDouble("money"); 41 42 System.out.println(id); 43 System.out.println(name); 44 System.out.println(money); 45 46 System.out.println("--------------"); 47 } 48 49 //7. 释放资源 50 rs.close(); 51 stmt.close(); 52 conn.close(); 53 }
案例
-
需求:查询account账户表数据,封装为Account对象中,并且存储到ArrayList集合中
-
代码实现
1 /** 2 * 查询account账户表数据,封装为Account对象中,并且存储到ArrayList集合中 3 * 1. 定义实体类Account 4 * 2. 查询数据,封装到Account对象中 5 * 3. 将Account对象存入ArrayList集合中 6 */ 7 @Test 8 public void testResultSet2() throws Exception { 9 //1. 注册驱动 10 //Class.forName("com.mysql.jdbc.Driver"); 11 //2. 获取连接:如果连接的是本机mysql并且端口是默认的 3306 可以简化书写 12 String url = "jdbc:mysql:///db1?useSSL=false"; 13 String username = "root"; 14 String password = "1234"; 15 Connection conn = DriverManager.getConnection(url, username, password); 16 17 //3. 定义sql 18 String sql = "select * from account"; 19 20 //4. 获取statement对象 21 Statement stmt = conn.createStatement(); 22 23 //5. 执行sql 24 ResultSet rs = stmt.executeQuery(sql); 25 26 // 创建集合 27 List<Account> list = new ArrayList<>(); 28 29 // 6.1 光标向下移动一行,并且判断当前行是否有数据 30 while (rs.next()){ 31 Account account = new Account(); 32 33 //6.2 获取数据 getXxx() 34 int id = rs.getInt("id"); 35 String name = rs.getString("name"); 36 double money = rs.getDouble("money"); 37 38 //赋值 39 account.setId(id); 40 account.setName(name); 41 account.setMoney(money); 42 43 // 存入集合 44 list.add(account); 45 } 46 47 System.out.println(list); 48 49 //7. 释放资源 50 rs.close(); 51 stmt.close(); 52 conn.close(); 53 }
PreparedStatement
PreparedStatement作用:预编译SQL语句并执行:预防SQL注入问题
SQL注入
SQL注入是通过操作输入来修改事先定义好的SQL语句,用以达到执行代码对服务器进行攻击的方法。
代码模拟SQL注入问题
1 @Test 2 public void testLogin() throws Exception { 3 //2. 获取连接:如果连接的是本机mysql并且端口是默认的 3306 可以简化书写 4 String url = "jdbc:mysql:///db1?useSSL=false"; 5 String username = "root"; 6 String password = "1234"; 7 Connection conn = DriverManager.getConnection(url, username, password); 8 9 // 接收用户输入 用户名和密码 10 String name = "xiaoming"; 11 String pwd = "' or '1' = '1"; 12 String sql = "select * from tb_user where username = '"+name+"' and password = '"+pwd+"'"; 13 // 获取Statement对象 14 Statement stmt = conn.createStatement(); 15 // 执行sql 16 ResultSet rs = stmt.executeQuery(sql); 17 // 判断登录是否成功 18 if(rs.next()){ 19 System.out.println("登录成功~"); 20 }else{ 21 System.out.println("登录失败~"); 22 } 23 24 //7. 释放资源 25 rs.close(); 26 stmt.close(); 27 conn.close(); 28 }
上面代码是将用户名和密码拼接到sql语句中,拼接后的sql语句如下
1 select * from tb_user where username = 'xiaoming' and password = ''or '1' = '1'
从上面语句可以看出条件 username = 'xiaoming' and password = ''
不管是否满足,而 or
后面的 '1' = '1'
是始终满足的,最终条件是成立的,就可以正常的进行登陆了。
PreparedStatement概述
PreparedStatement作用:预编译SQL语句并执行:预防SQL注入问题
-
获取 PreparedStatement 对象
1 // SQL语句中的参数值,使用?占位符替代 2 String sql = "select * from user where username = ? and password = ?"; 3 // 通过Connection对象获取,并传入对应的sql语句 4 PreparedStatement pstmt = conn.prepareStatement(sql);
-
设置参数值
上面的sql语句中参数使用 ? 进行占位
PreparedStatement对象:setXxx(参数1,参数2):给 ? 赋值
-
setXxx:数据类型 ; 如 setInt (参数1,参数2)
-
参数:
-
参数1: ?的位置编号,从1 开始
-
参数2: ?的值
-
-
-
执行SQL语句
executeUpdate(); 执行DDL语句和DML语句
executeQuery(); 执行DQL语句
注意:调用这两个方法时不需要传递SQL语句,因为获取SQL语句执行对象时已经对SQL语句进行预编译了。
使用PreparedStatement改进
1 @Test 2 public void testPreparedStatement() throws Exception { 3 //2. 获取连接:如果连接的是本机mysql并且端口是默认的 3306 可以简化书写 4 String url = "jdbc:mysql:///db1?useSSL=false"; 5 String username = "root"; 6 String password = "1234"; 7 Connection conn = DriverManager.getConnection(url, username, password); 8 9 // 接收用户输入 用户名和密码 10 String name = "zhangsan"; 11 String pwd = "' or '1' = '1"; 12 13 // 定义sql 14 String sql = "select * from tb_user where username = ? and password = ?"; 15 // 获取pstmt对象 16 PreparedStatement pstmt = conn.prepareStatement(sql); 17 // 设置?的值 18 pstmt.setString(1,name); 19 pstmt.setString(2,pwd); 20 // 执行sql 不用带参数了 21 ResultSet rs = pstmt.executeQuery(); 22 // 判断登录是否成功 23 if(rs.next()){ 24 System.out.println("登录成功~"); 25 }else{ 26 System.out.println("登录失败~"); 27 } 28 //7. 释放资源 29 rs.close(); 30 pstmt.close(); 31 conn.close(); 32 }
执行上面语句就可以发现不会出现SQL注入漏洞问题了。那么PreparedStatement又是如何解决的呢?它是将特殊字符进行了转义,转义的SQL如下:
1 select * from tb_user where username = 'zhangsan' and password = '\'or \'1\' = \'1'
PreparedStatement原理
PreparedStatement 好处:
-
预编译SQL,性能更高 (把sql语句传递给PreparedStatement 时就会进行预编译,不会等到程序执行在预编译)
-
防止SQL注入:将敏感字符进行转义
Java代码操作数据库流程如图所示:
-
将sql语句发送到MySQL服务器端
-
MySQL服务端会对sql语句进行如下操作
-
检查SQL语句
检查SQL语句的语法是否正确。
-
编译SQL语句。将SQL语句编译成可执行的函数。
检查SQL和编译SQL花费的时间比执行SQL的时间还要长。如果我们只是重新设置参数,那么检查SQL语句和编译SQL语句将不需要重复执行(只执行一次)。这样就提高了性能。
-
执行SQL语句
-
接下来我们通过查询日志来看一下原理。
-
开启预编译功能
在代码中编写url时需要加上以下参数。而我们之前根本就没有开启预编译功能,只是解决了SQL注入漏洞。
1 useServerPrepStmts=true
-
配置MySQL执行日志(重启mysql服务后生效)
在mysql配置文件(my.ini)中添加如下配置
1 log-output=FILE 2 general-log=1 3 general_log_file="D:\mysql.log" 4 slow-query-log=1 5 slow_query_log_file="D:\mysql_slow.log" 6 long_query_time=2
-
java测试代码如下:
1 /** 2 * PreparedStatement原理 3 * @throws Exception 4 */ 5 @Test 6 public void testPreparedStatement2() throws Exception { 7 8 //2. 获取连接:如果连接的是本机mysql并且端口是默认的 3306 可以简化书写 9 // useServerPrepStmts=true 参数开启预编译功能 10 String url = "jdbc:mysql:///db1?useSSL=false&useServerPrepStmts=true"; 11 String username = "root"; 12 String password = "1234"; 13 Connection conn = DriverManager.getConnection(url, username, password); 14 15 // 接收用户输入 用户名和密码 16 String name = "zhangsan"; 17 String pwd = "' or '1' = '1"; 18 19 // 定义sql 20 String sql = "select * from tb_user where username = ? and password = ?"; 21 22 // 获取pstmt对象 23 PreparedStatement pstmt = conn.prepareStatement(sql); 24 25 // 设置?的值 26 pstmt.setString(1,name); 27 pstmt.setString(2,pwd); 28 ResultSet rs = null; 29 // 执行sql 30 rs = pstmt.executeQuery(); 31 32 //只会进行一次预编译,下边语句不会预编译 33 // 设置?的值 34 pstmt.setString(1,"aaa"); 35 pstmt.setString(2,"bbb"); 36 // 执行sql 37 rs = pstmt.executeQuery(); 38 39 // 判断登录是否成功 40 if(rs.next()){ 41 System.out.println("登录成功~"); 42 }else{ 43 System.out.println("登录失败~"); 44 } 45 46 //7. 释放资源 47 rs.close(); 48 pstmt.close(); 49 conn.close(); 50 }
-
执行SQL语句,查看
D:\mysql.log
日志如下:上图中第三行中的
Prepare
是对SQL语句进行预编译(需要设置useServerPrepStmts=true 参数开启预编译功能)。第四行和第五行是执行了两次SQL语句,而第二次执行前并没有对SQL进行预编译。
小结:
-
在获取PreparedStatement对象时,将sql语句发送给mysql服务器进行检查,编译(这些步骤很耗时)
-
执行时就不用再进行这些步骤了,速度更快
-
如果sql模板一样,则只需要进行一次检查、编译
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 无需6万激活码!GitHub神秘组织3小时极速复刻Manus,手把手教你使用OpenManus搭建本
· C#/.NET/.NET Core优秀项目和框架2025年2月简报
· 葡萄城 AI 搜索升级:DeepSeek 加持,客户体验更智能
· 什么是nginx的强缓存和协商缓存
· 一文读懂知识蒸馏