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)

大家都在看

  • 必应咋想的

    首页里弄了个阴森的图片,下面有个山洞,里面有白衣女鬼飘过,还有背景音乐。 看右下角的介绍里有名字叫:万圣节之夜在黑暗树篱 Original: https://www.cnblogs…

    数据库 2023年6月9日
    0103
  • md5解密异常

    javax.crypto.BadPaddingException: Given final block not properly paddedat com.sun.crypto.p…

    数据库 2023年6月11日
    0101
  • 红黑树

    2-3-4树 JAVA技术交流群:737698533 定义 所有的叶子节点都拥有相同的深度 节点只能是2-节点,3-节点,或者4-节点 2节点 包含一个元素的节点,有两个子节点 3…

    数据库 2023年6月16日
    070
  • 【数据结构】跳表

    一、基本概念 1.1 定义 跳表(SkipList):增加了向前指针的链表叫做指针。跳表全称叫做跳跃表,简称跳表。跳表是一个随机化的数据结构,实质是一种可以进行二分查找的有序链表。…

    数据库 2023年6月11日
    094
  • Rabbitmq从安装到简单入门

    1:Rabbitmq是什么? RabbitMQ是实现了高级消息队列协议(AMQP)的开源消息代理软件(亦称面向消息的中间件)。 它由以高性能、健壮以及可伸缩性出名的 Erlang …

    数据库 2023年6月6日
    086
  • Java中的线程安全与线程同步

    1.为什么需要线程同步 什么是线程安全:指在 被多个线程访问时,程序可以 持续进行正确的处理。 1.1.线程安全问题 案例:通过抢优惠例子说明线程安全问题 public class…

    数据库 2023年6月6日
    098
  • 三分钟图解事务隔离级别,看一遍就懂

    前文说过,”锁” 是数据库系统区别于文件系统的一个关键特性,其对象是 事务,用来锁定的是数据库中的对象,如表、页、行等。锁确实提高了并发性,但是却不可避免地…

    数据库 2023年5月24日
    0122
  • Win10系统-接口自动化测试持续集成

    使用工具:jdk+jmeter+Ant+jenkins jdk-1.8.0_241版本【安装参考链接:https://blog.51cto.com/u_15463439/52268…

    数据库 2023年6月14日
    0100
  • 限流常见方案

    限流常见方案 我歌月徘徊,我舞影零乱。醒时相交欢,醉后各分散。 一、限流思路 常见的系统服务限流模式有:熔断、服务降级、延迟处理和特殊处理四种。 1、熔断 将熔断措施嵌入到系统设计…

    数据库 2023年6月14日
    094
  • tomcat服务的原理和使用

    一、tomcat与服务器的关系 服务器是指安装了服务器软件的计算机,而服务器软件是指接收用户请求,处理请求,返回请求结果的程序,常见的服务器软件有tomcat、iis等,也称为we…

    数据库 2023年6月14日
    0120
  • 没有 Cgroups,就没有 Docker

    Cgroups 是什么? Cgroups 是 control groups 的缩写,是 Linux 内核提供的一种可以 限制、 记录、 隔离进程组(progress groups)…

    数据库 2023年6月6日
    0116
  • Nginx常见问题

    基于端口的虚拟机: 使用端口来区分,浏览器使用域名或ip地址:端口号 访问。 注意:~ 代表自己输入的英文字母。 16、location正则案例 1 #优先级1,精确匹配,根路径 …

    数据库 2023年6月14日
    0125
  • mapreduce统计单词个数

    WordCount类代码: import org.apache.hadoop.conf.Configuration; import org.apache.hadoop.fs.Pat…

    数据库 2023年6月11日
    095
  • 写给所有程序员的对象的一封信

    因为本人有一枚可爱的老婆,她经常有很多奇怪的问题(我承认其实是我老想跟她分享),但是有些问题需要有一定的理论支撑,所以我就打算在这里一并告诉她。就是一些关于编程的前置知识的汇总,如…

    数据库 2023年6月14日
    078
  • 使用clipboard.js复制文字+图片到微信后图片不显示问题处理

    使用clipboard.js复制文字 +图片,粘贴到微信不显示图片,而QQ可以。 解决方案: 图片链接使用http,不要使用https。 使用clipboard.js实现复制功能 …

    数据库 2023年6月14日
    0115
  • MySql用户与权限控制

    MySql用户与权限控制 — 刷新权限命令 # — 刷新mysql权限命令 flush privileges; 用户管理 1、查看用户 #查看用户 USE mysql…

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