JDBC

JDBC

一、JDBC概述

什么是JDBC?

JDBC

JDBC 是使用 Java 语言操作关系型数据库的一套 API。这套 API 是交由不同的数据库厂商实现的。我们利用 JDBC 编写操作数据库的代码,真正执行的是各个数据库的 实现类驱动)。

全称:(Java DataBase Connectivity)Java 数据库连接。

JDBC的好处

  • 面向接口编程,屏蔽实现上的差异。
  • 一套 Java 代码操作不同数据库。、

二、使用JDBC

环境配置


    mysql
    mysql-connector-java
    8.0.29

编码

步骤

  1. 引入驱动并注册
  2. 获取连接
  3. 定义SQL
  4. 获取执行SQL对象
  5. 执行SQL
  6. 处理返回结果
  7. 释放资源

代码实现

public static void demo(){
    Connection conn = null;
    Statement state = null;
    try {
        // 1.注册驱动
        Class.forName("com.mysql.cj.jdbc.Driver");
        // 2.获取连接
        conn = DriverManager.getConnection("jdbc:mysql://localhost:3306/test", "root", "root");
        // 3.定义SQL
        String sql = "select * from user";
        // 4.获取Statement对象
        state = conn.createStatement();
        // 5.执行SQL
        ResultSet resultSet = state.executeQuery(sql);
        // 6.处理返回结果
        System.out.println(resultSet);
    } catch (ClassNotFoundException | SQLException e) {
        e.printStackTrace();
    } finally {
        // 7.关闭资源
        if (state != null) {
            try {
                state.close();
            } catch (SQLException e) {
                e.printStackTrace();
            }
        }
        if (conn != null) {
            try {
                conn.close();
            } catch (SQLException e) {
                e.printStackTrace();
            }
        }
    }
}

注意

  • 最晚获得的资源最先关闭。
  • Mysql8 之前的驱动类全类名为 com.mysql.jdbc.DriverMysql8之后则为 com.mysql.cj.jdbc.Driver

三、JDBC API

DriverManager

作用

  • 注册驱动
  • 获取数据库连接

这两个作用对应着两个方法:

方法 作用 getConnection(String url, String user, String password) 通过给定的URL与数据库建立连接,并返回连接对象 registerDriver(Driver driver) 注册给定的驱动 Driver

Mysql 驱动底层通过 静态代码块调用了 DriverManagerregisterDriver方法,所以我们不必去显式的进行驱动的注册,它已经帮我们注册好了。

public class Driver extends NonRegisteringDriver implements java.sql.Driver {
    public Driver() throws SQLException {
    }

    static {
        try {
            DriverManager.registerDriver(new Driver());
        } catch (SQLException var1) {
            throw new RuntimeException("Can't register driver!");
        }
    }
}

注意

如果导入 MySQL5 之后的驱动 jar包,可以不用写 Class.forName("com.mysql.cj.jdbc.Driver");,Driver 会自动读取 java.sql.Driver文件中的驱动类全限定名,进行驱动的注册。

JDBC

Connection

作用

  • 获取执行SQl的对象
  • 管理事务

执行SQL的对象

  • Statement 普通的执行对象,将获得的字段拼接在SQl语句上,然后在执行编译,有SQl注入攻击的风险。
  • PerparedStatement 对SQL进行 预编译的执行对象,先对SQl语句进行编译,再用字段将 ?替换掉。
  • CallableStatement 执行存储过程的对象。

事务管理

方法 说明 setAutoCommit(boolean) 开启事务。true:自动提交事务,false:开启手动提交事务 commit() 提交事务 rollback() 回滚事务

try {
    // 开启事务
    conn.setAutoCommit(false);
    // 执行操作一
    int i = state.executeUpdate(sql1);
    System.out.println(i);
    // 制造异常
    int j = 1/0;
    // 执行操作二
    int i1 = state.executeUpdate(sql2);
    System.out.println(i1);
    // 提交事务
    conn.commit();
} catch (Exception e) {
    // 事务回滚
    conn.rollback();
    e.printStackTrace();
}

Statement

作用

执行 sql 语句。

方法

方法 说明 int executeUpdate

(String sql) 执行给定的SQL语句,这可能是 INSERT UPDATE

,或 DELETE

语句,或者不返回任何内容,如SQL DDL语句的SQL语句。并返回受影响的行数。 ResultSet executeQuery

(String sql) 执行给定的SQL语句,该语句返回单个 ResultSet

对象。

ResultSet

作用

封装了查询语句的执行结果, Statement 或者 PreparedStatement 通过执行 executeQuery 方法返回 ResultSet 对象。

方法

方法 说明 getXxx() 获取每行中各个数据,可以传入 列号

(从1开始) 或者 列名

next() 向下移动指针,判断该行是否有数据。

JDBC
// 6.处理返回结果
while (resultSet.next()) {
    String name = resultSet.getString("name");
    System.out.println(name);
    String password = resultSet.getString(3);
    System.out.println(password);
}

PerparedStatement

作用

执行 sql 语句,但是它会对 SQl 语句进行 预编译,可以防止 SQl 注入攻击。并且 PreparedStatement 继承自 Statement

SQl注入

在编译之前利用 SQl 语句的拼接,输入特殊字符串修改定义好的 SQL 语句,改变 SQL 语句的语义。

一个简单的 SQL 注入演示:

(比如是有一个用户登录的场景,在 DAO 层中进行数据库操作)

String sql = "select * from user where username='"+admin+"' and password='"+'or'1'='1+"'";

用户密码一看就不正经,但是可以看一下拼接好的 SQL 语句是怎样的:

select * from user where username='admin' and password=''or'1'='1'

password 变成了空字符串,在 SQL 语句的尾部 拼接了 or '1'='1' 。username 和 password 查询为 false,但是接着又 or 一个 恒等式,整体结果就变成了 true,将会查询出所有结果。

预编译

PreparedStatement 的使用步骤:

// 3.定义SQL
String sql = "select * from user where name=? and password=?";
// 4.获取PreparedStatement对象
PreparedStatement statement = conn.prepareStatement(sql);
// 4.5替换占位符
statement.setString(1,"黑夫");
statement.setString(2,"123");
// 5.执行SQL
ResultSet resultSet = statement.executeQuery();

在定义 SQL 语句时,SQL 中未知字段使用 ? 进行占位。

可以看到在获取 PreparedStatement 对象的同时,就需要将 SQL 传入,这时会将 SQL 语句发送给 mysql 服务器,进行检查语法、编译等。

为什么能达到防止 SQL 注入的效果?

SQL 语句在执行之前已经进行了编译,它的结构已经被固定下来,当运行时动态地把参数传给 PreprareStatement 时,即使参数里有敏感字符如 or ‘1=1’数据库会将它作为一个参数一个字段的 属性值来处理而不会作为一个 SQL 指令。而且 PreparedStatement 在用具体字段替换占位符时,会对特殊字符进行 转义,比如单引号 ' 会转义为 \'。如此,可以有效的防止 SQL 注入。

可以在 url 后面加上命令开启预编译: useServerPrepStmts=true。不写这句,仍会对特殊字符进行转义。

jdbc:mysql://localhost:3306/test?useSSL=false&useServerPrepStmts=true

useSSL=false 表示不进行安全连接。

预编译的好处

  • 防止 SQL 注入
  • 一次编译、多次运行,省去了解析优化等过程 在执行 SQL 时,很多 SQL 语句都是结构相同的,只是个别字段不同,如果对每一条 SQL 进行编译,那么运行速度将大大降低。预编译时会将编译过后的SQL语句进行缓存,当有结构相同的 SQL 语句需要执行时,只需用属性值替换掉占位符即可,不需要重新 词法语义解析语句优化制定执行计划编译等,从而提高效率。

关于更多 PreparedStatement 的了解,看这篇博文:

预编译语句(Prepared Statements)介绍,以MySQL为例

四、数据库连接池

简介

数据库连接池是负责 分配、管理和释放数据库连接的容器,它允许应用程序 重复使用一个现有的数据库连接,而不是再重新建立一个。

对于多用户的应用,如果为每一个用户都创建一个数据库连接,毫无疑问是对资源的浪费,同时也浪费时间。我们应当在系统加载的时候就将数据库资源创建好,不推荐运行时再去开启资源。做到对资源的统一管理,从而避免资源浪费,提升响应速度。

使用

SUN 公司提供了数据库连接池的标准接口 DataSource 由第三方组织进行实现。

推荐使用 Druid 数据库连接池技术。

环境搭建


    com.alibaba
    druid
    1.2.11

在resources目录下新建 druid.properties 配置文件

driverClassName = com.mysql.cj.jdbc.Driver
url = jdbc:mysql://localhost:3306/test
username = root
password = root
初始化连接数
initialSize = 5
最大连接数
maxActive = 10
最大等待时间
maxWait = 3000

maxWait 表示如果连接池中没有空闲连接的最大等待时间,超过时间则会抛出异常。

更多配置参考:Configuration reference

获取连接

// 1.加载配置文件
Properties properties = new Properties();
properties.load(ClassLoader.getSystemResourceAsStream("druid.properties"));
// 2.获取连接
DataSource dataSource = DruidDataSourceFactory.createDataSource(properties);
conn = dataSource.getConnection();

Original: https://www.cnblogs.com/suwuji/p/16631067.html
Author: 苏无及
Title: JDBC

原创文章受到原创版权保护。转载请注明出处:https://www.johngo689.com/611575/

转载文章受原作者版权保护。转载请注明原作者出处!

(0)

大家都在看

  • PHP获取前一天,前一个月,前半年,前一年的时间戳

    // 获取前一秒 strtotime("-1 seconds"); // 获取前一分钟 strtotime("-1 minute"); //…

    数据库 2023年6月14日
    0122
  • Springboot打包部署项目

    这里打包的是jar项目,也就是没有webapp目录,通过maven打包插件打包发布到服务器 废话不多少直接开撸废话不多少直接开撸废话不多少直接开撸废话不多少直接开撸废话不多少直接开…

    数据库 2023年6月6日
    075
  • 我的第一次校招

    2018-09-26 23:40:03 虽然是第一次参加,但这次的笔试完成结果让我不是很满意,因为有几道超简单的字符串编程没有做,忘了或者是想复杂了,还有一些概念题不是很清楚,自己…

    数据库 2023年6月16日
    0108
  • Mysql 实现数据库读写分离

    一、Amoeba 是什么 Amoeba(变形虫)项目,专注 分布式数据库 proxy 开发。座落与Client、DB Server(s)之间。对客户端透明。具有负载均衡、高可用性、…

    数据库 2023年6月14日
    071
  • 草图?不管黑猫白猫,能把你的设计理念讲清楚才行

    我在日常工作中,经常要参加一些技术活动,或被拉去参加一些需求会或运营会,时间比较分散。 上周在参加一个代码评审时,发现程序上该复用的没有复用,却写了两份逻辑几乎相同的代码。另外,还…

    数据库 2023年6月9日
    070
  • Mysql 数据恢复逻辑 基于binlog redolog undolog

    注:文中有个易混淆的地方”事务” sql事务,即每次数据库操作生成的事务,这个事务trx_id只在undolog里存储,因为MVVC需要记录修改的事务id,…

    数据库 2023年5月24日
    082
  • nginx反向代理proxy_pass url后加/和不加/的区别

    在nginx中配置proxy_pass反向代理时,当在后面的url加上了/,相当于是绝对根路径,则nginx不会把location中匹配的路径部分代理走;如果没有/,则会把匹配的路…

    数据库 2023年6月6日
    0115
  • mysql扫描全表更新状态部分失败

    一直以为mysql是按照主键排序的,实则排序和主键没有关系(不使用 order by 子句)。 然后从 stackoverflow 上查了一下,找到了以下的回答: 没有默认的排序顺…

    数据库 2023年5月24日
    074
  • java crm 进销存 模块设计方案

    主页: 记录总进货额、总销售额、总销售利润、30天内销售金额统计总客户数、30天新增客户数图表统计(折线图和柱状图)展示从当前月份开始前12个月销售情况1. 客户资料:记录客户资料…

    数据库 2023年6月6日
    086
  • Python日志模块封装

    一、先上结论 对 Python logging模块进行二次封装 -*- coding:utf-8 -*- 作者:IT小学生蔡坨坨 博客:caituotuo.top 时间:2022/…

    数据库 2023年6月11日
    084
  • springcloud~nacos通过@refreshScope进行配置热更新

    配置类 @Data @ConfigurationProperties("auth") public class AuthProperties { private…

    数据库 2023年6月6日
    069
  • 事物的隔离性和MVCC

    事物的隔离性 mysql的服务端是支持多个客户端同时与之连接的,每个客户端可能还并发了好几个连接,所以mysql是需要同时处理很多事情的,每一件独立的事情就叫做事务。我们知道事务有…

    数据库 2023年5月24日
    084
  • Python环境安装

    一、下载地址: Python:Download Python | Python.org PyCharm:Download PyCharm: Python IDE for Profe…

    数据库 2023年6月14日
    062
  • Spring Security实现统一登录与权限控制

    1 项目介绍 最开始是一个单体应用,所有功能模块都写在一个项目里,后来觉得项目越来越大,于是决定把一些功能拆分出去,形成一个一个独立的微服务,于是就有个问题了,登录、退出、权限控制…

    数据库 2023年6月14日
    093
  • Redis——数据操作(3)

    2022-09-22 (4)set操作 ①添加操作(sadd): 例: ②查看操作(smembers):例: ③移除操作(srem):例: (5)zset(有序集合而且里面的元素是…

    数据库 2023年6月14日
    0121
  • 分享一直在维护简单实用高效的C++Socket框架Swa-server(开源+源码)

    Swa-server 开源框架* 适用于中小型游戏,如:养成、RPG、棋牌等;应用软件,如:聊天室等* 已经封套好底层socket管理,sql请求处理、数据加密解密* 拿来即可开工…

    数据库 2023年6月14日
    088
亲爱的 Coder【最近整理,可免费获取】👉 最新必读书单  | 👏 面试题下载  | 🌎 免费的AI知识星球