JDBC

JDBC

一、JDBC概述

什么是JDBC?

JDBC

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

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

JDBC的好处

  • 面向接口的编程,屏蔽实现差异。
    [En]

    Interface-oriented programming to shield implementation differences.*

  • 一套 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();            }        }    }}

注意

  • 最新资源率先关闭
    [En]

    the latest resources are the first to be closed.*

  • Mysql8 之前的驱动类全类名为 com.mysql.jdbc.DriverMysql8之后则为 com.mysql.cj.jdbc.Driver

三、JDBC API

DriverManager

作用

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

这两个操作对应于两种方法:

[En]

These two actions correspond to two methods:

方法 作用 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为例

四、数据库连接池

简介

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

对于多用户应用来说,为每个用户创建一个数据库连接无疑是浪费资源和时间。我们应该在系统加载时创建数据库资源,不建议在运行时打开资源。实现资源统一管理,避免资源浪费,提高响应速度。

[En]

For multi-user applications, it is undoubtedly a waste of resources and time to create a database connection for each user. We should create the database resources when the system is loaded, and it is not recommended to open the resources at run time. Achieve the unified management of resources, so as to avoid the waste of resources and improve the response speed.

使用

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/504926/

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

(0)

大家都在看

  • 当mysql表从压缩表变成普通表会发生什么

    本文章做了把mysql表从压缩表过渡到普通表的实验过程,看看压缩表变成普通表会发生什么?本文针对mysql5.7和mysql8分别进行了实验。 1、什么是表压缩 在将压缩表引入普通…

    数据库 2023年5月24日
    073
  • 最新Mysql大厂面试必会的34问题

    1、mysql的隔离级别 四种隔离级别: READ-UNCOMMITTED(读取未提交): 最低的隔离级别,允许读取尚未提交的数据变更, 可能会导致脏读、幻读或不可重复读 。 RE…

    数据库 2023年5月24日
    087
  • mysql常用操作汇总

    工作中经常用会遇到这种情况,可以访问mysql所在的服务器,但是服务器端口不对外暴露(通常因为安全原因)。这时,操作数据库只能通过命令行和 mysql client窗口来实现。我对…

    数据库 2023年5月24日
    086
  • 粗粒度服务的执行时间统计算法实现及问题推广

    3.1 算法一:逐一合并算法 3.1.1 算法描述 如图3.我们先将所有的时间片段按照起始时刻排序,后面的处理过程即依次判定相邻两个时间片段的三种关系,并将统计时间进行合并,合并的…

    数据库 2023年6月14日
    0115
  • 如何把返回的datatable按某个字段 排序 升序或者降序

    如何把返回的datatable按某个字段 排序 升序或者降序 DataTable dtdata = GetXmlData(doc, “DetailList”…

    数据库 2023年6月9日
    082
  • SQL语句实战学习

    参考:https://zhuanlan.zhihu.com/p/38354000再次感谢作者的整理!! 1.数据已提前准备好了,已知有如下4张表:学生表:student 成绩表:s…

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

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

    数据库 2023年6月14日
    091
  • day39-网络编程01

    Java网络编程01 1.网络相关的概念 1.1网络通信和网络 *网络通信 概念:两台设备之间通过网络实现数据传输 网络通信:将数据通过网络从一台设备传输到另一台设备 java.n…

    数据库 2023年6月11日
    0109
  • 操作系统(学习笔记)

    操作系统(学习笔记) PCB=process control block=进程控制块,用于存储进程相关信息,以便进程切换; GDT=global descriptor table=…

    数据库 2023年6月14日
    085
  • MySQL45讲之优化器选错索引

    前言 本文简要介绍了优化器选择索引的依据,以及如何人为地引导优化器选择较好的执行方案。 [En] This paper briefly introduces the basis f…

    数据库 2023年5月24日
    0134
  • Linux

    1、关机命令 命令 说明 sync 将数据由内存同步到硬盘中 shutdown 关机 shutdown -h 10 10分钟后关机 shutdown -h now 立马关机 shu…

    数据库 2023年6月16日
    0112
  • Java实现负载均衡算法–轮询和加权轮询

    1.普通轮询算法 轮询(Round Robin,RR)是依次将用户的访问请求,按循环顺序分配到web服务节点上,从1开始到最后一台服务器节点结束,然后再开始新一轮的循环。这种算法简…

    数据库 2023年6月6日
    0106
  • 977.有序数组的平方

    给你一个按 非递减顺序 排序的整数数组 nums,返回 每个数字的平方 组成的新数组,要求也按 非递减顺序 排序。 示例 1: 输入:nums = [-4,-1,0,3,10]输出…

    数据库 2023年6月16日
    099
  • pytest中pytest_cache文件夹作用

    跑自动化时经常会出现这样一个情况,一轮自动化跑完后零星出现了几个失败case,无法断定失败的原因,所以需要重新跑一下失败的case去debug,那我们要做的是就去修改脚本把那几个c…

    数据库 2023年6月11日
    065
  • 在浏览器中Django项目的静态文件打不开的一个原因

    2022-09-27 问题描述: 编写Django代码时,设置了一个”static”文件夹,在里面放置了一张图片。在”setting&#8221…

    数据库 2023年6月14日
    0110
  • Redis学习(1)—Redis概述

    什么是NoSQL NoSQL:Not Only SQL,意思不仅仅是SQL,它是属于 非关系型数据库。那什么是关系型数据库?数据结构是一种有行有列的数据库。 NoSQL数据库是为了…

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