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 :用户名

    • poassword :密码

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概述

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模板一样,则只需要进行一次检查、编译

posted @   暴躁C语言  阅读(26)  评论(0编辑  收藏  举报
相关博文:
阅读排行:
· 无需6万激活码!GitHub神秘组织3小时极速复刻Manus,手把手教你使用OpenManus搭建本
· C#/.NET/.NET Core优秀项目和框架2025年2月简报
· 葡萄城 AI 搜索升级:DeepSeek 加持,客户体验更智能
· 什么是nginx的强缓存和协商缓存
· 一文读懂知识蒸馏
点击右上角即可分享
微信分享提示