JDBC 编程

目录

JDBC 是什么

JDBC 的工作原理

JDBC 的使用

引入驱动

使用

常用接口和类

Connection

Statement

ResultSet

使用总结


JDBC 是什么

JDBC(Java Database Connectivity):Java数据库连接,是一种用于执行 SQL 语句的Java API,是 Java 数据库连接规范,为 Java 开发人员操作数据库提供了一个标准的 API,可以为多种关系数据库提供统一访问

数据库是一个类别,其中包括 MySQL、Oracle、sql server 等,这些数据库提供的 API 有所不同,也就增加了我们的学习成本

而 JDBC 为实现数据库操作提供了一套标准接口,让 Java 应用程序能够与各种数据库系统交互,而不需要关键具体数据库的实现细节

JDBC 的工作原理

JDBC 为多种关系数据库提供了统一访问方式,主要包含一些通用的接口类

JDBC 访问数据库的层次结构

JDBC的优势:

Java语言操作数据库时完全面向抽象接口编程

开发数据库应用不用限定特定数据库厂商的API

程序的可移植性大大增强

JDBC 的使用

引入驱动

在使用 JDBC 之前,需要引入 JDBC 驱动包

在仓库中进行下载:

Maven Repository: Search/Browse/Explore (mvnrepository.com)

搜索mysql:

 其中:

Connector/J 中存放的是更高版本的驱动包

Connector Java 中存放的是低版本的驱动包

根据服务器的版本选择对应驱动包:

选择对应版本的驱动包,并下载:

接着,我们将 jar 包引入到项目中

我们先创建一个新项目,并创建一个目录,并将 jar 包拷贝到这个目录中:

再右键这个新目录,add as library: 

接下来,我们就可以编写代码了

使用

首先,需要创建数据源(数据源描述了需要操作的数据库在哪里),设置数据库连接的 url、用户名 和 密码:

DataSource dataSource = new MysqlDataSource();
((MysqlDataSource) dataSource).setUrl("jdbc:mysql://127.0.0.1:3306/test?characterEncoding=utf8&useSSL=false");
((MysqlDataSource) dataSource).setUser("root");
((MysqlDataSource) dataSource).setPassword("123456");

其中,MySQL 数据连接的 url 参数格式为:

jdbc:mysql://服务器地址:端口/数据库名?参数名=参数值

charsetEncoding:设置字符集和编码方式

useSSL:数据是否进行加密传输

接着,和数据库服务器建立连接:

Connection connection = dataSource.getConnection();

在建立连接的过程中可能会出错(抛出异常),如 url 不正确、用户名有误、密码错误等,需要我们进行处理

然后,就可以构造 SQL 语句并执行了

        // 构造 SQL
        String sql = "insert into class values (4, '四班')";
        PreparedStatement statement = connection.prepareStatement(sql);

        // 执行 SQL
        int result = statement.executeUpdate();
        // 查看执行结果
        System.out.println(result);

最后,别忘记释放资源

statement.close();
connection.close();

上述完整代码:

public class Test {
    public static void main(String[] args) throws SQLException {
        // 构造 DataSourse
        DataSource dataSource = new MysqlDataSource();
        ((MysqlDataSource) dataSource).setUrl("jdbc:mysql://127.0.0.1:3306/test?characterEncoding=utf8&useSSL=false");
        ((MysqlDataSource) dataSource).setUser("root");
        ((MysqlDataSource) dataSource).setPassword("123456");

        // 建立连接
        Connection connection = dataSource.getConnection();

        // 构造 SQL
        String sql = "insert into class values (4, '四班')";
        PreparedStatement statement = connection.prepareStatement(sql);

        // 执行 SQL
        int result = statement.executeUpdate();
        // 查看执行结果
        System.out.println(result);

        // 执行完毕,释放资源
        statement.close();
        connection.close();
    }
}

总结一下 JDBC 的使用步骤:

1. 创建数据库连接 Connection

2. 创建操作命令

3. 执行SQL

4. 若进行查询操作,处理结果集 ResultSet

5. 释放资源

在通过 JDBC 对数据库进行操作时,需要使用其提供的 API 进行统一处理,因此,我们来进一步学习 JDBC API 

常用接口和类

Connection

Connection 接口用于与数据库建立连接,由数据库提供,获取 Connection 对象通常有两种方式:

(1)通过 DriverManager(驱动管理类)的静态方法获取

// 数据库连接信息
String url = "jdbc:mysql://127.0.0.1:3306/test?characterEncoding=utf8&useSSL=false";
String user = "root";
String password = "123456";
// 加载 JDBC 驱动
Class.forName("com.mysql.jdbc.Driver");
// 创建数据库连接
Connection connection = DriverManager.getConnection(url, user, password);

(2)通过 DataSource(数据源)对象获取

// 构造 DataSourse
DataSource dataSource = new MysqlDataSource();
((MysqlDataSource) dataSource).setUrl("jdbc:mysql://127.0.0.1:3306/test?characterEncoding=utf8&useSSL=false");
((MysqlDataSource) dataSource).setUser("root");
((MysqlDataSource) dataSource).setPassword("phantom5170&");
// 建立连接
Connection connection = dataSource.getConnection();

上述两种方式的区别为:

使用 DriverManager 类获取的 Connection 连接,是无法重复利用的,每次使用完后释放资源时,通过 connection.close() 关闭物理连接

DataSource 提供连接池支持,连接池在初始化创建时将创建一定数量的数据库连接,这些连接是可以复用的,每次使用完数据库连接,释放资源调用 connection.close() 都是将 Connection 连接对象回收

Statement

Statement 对象主要是将 SQL 语句发送到数据库中,JDBC API 中主要提供了三种 Statement 对象

Statement:通常用于执行不带参数的简单 SQL 语句

PreparedStatement:用于执行带或不带参数的SQL语句,SQL 语句会进行预编译,PreparedStatement 的执行速度快于 Statement

CallableStatement:用于执行数据库存储过程中的调用

其中,PreparedStatement 是最常用的,接下来,我们就来继续学习 PreparedStatement

什么是预编译?

当数据库服务器收到一条SQL语句之后,会:

先解析语法和语义,校验 SQL 语句是否正确

再优化 SQL 制定执行计划

执行并返回结果

 若一条 SQL 执行上述流程,就称为  Immediate Statements(即时SQL)

而 预编译 SQL 则会在编译一次之后将 SQL 语句缓存起来,后面再执行这条语句时,就不会再进行编译,省去了解析优化等过程

预编译 SQL 可以提高执行效率,因为 SQL 语句只需编译一次,然后可以多次执行,避免了重复编译的开销。预编译 SQL 通常用于需要频繁执行的 SQL 语句,如重复执行的查询或更新操作

即:

Immediate Statements(即时SQL):每次执行 SQL 语句时,数据库管理系统都会即时地进行语法分析、语义分析、优化和执行,而不是提前进行编译和优化。每次执行即时 SQL 语句都会有一定的性能开销,因为需要进行完整的编译和优化过程。

预编译 SQL(Prepared SQL):在预编译阶段,数据库管理系统(DBMS)会对 SQL 语句进行编译,生成执行计划,并将其存储在数据库中,而不是在每次执行时重新编译。在执行阶段,应用程序发送带有占位符的预编译 SQL 语句给 DBMS,DBMS 根据预编译的执行计划执行 SQL,并将实际参数传递给占位符,以生成最终的查询结果

 如何向 SQL 语句传递参数?

通常使用 字符串 来构造 SQL 语句,如 String sql = "insert into class values (4, '四班')" 此时语句中的参数是固定的,但是若我们需要让用户来指定参数,此时,该如何传递参数呢?

PreparedStatement 可以使用 ? 作为占位符,后续再进行参数的指定:

// 构造 SQL
String sql = "insert into class values (?, ?)";
// 将 string sql 转换为语句对象
PreparedStatement statement = connection.prepareStatement(sql);
// 获取输入的参数
Scanner scanner = new Scanner(System.in);
System.out.print("请输入班级id:>");
int classId = scanner.nextInt();
System.out.print("请输入班级名称:>");
String className = scanner.next();
// 设置 sql 参数
statement.setInt(1, classId);
statement.setString(2, className);

先使用 ? 对要进行传递的参数进行占位,后续再设置参数的值

paramenterIndex:参数下标

x:参数的值

PreparedStatement 中主要使用以下两种执行SQL的方法:

executeQuery():通常用于执行 select 语句,执行后返回一个结果集(ResultSet)

executeUpdate():通常用于执行 update、insert delete 语句,返回受影响的行数

我们对执行上述构造的 sql:

// 执行 SQL
int result = statement.executeUpdate();
// 查看执行结果
System.out.println(result);

返回结果为1,表示执行的语句影响了一行,说明插入操作成功

ResultSet

ResultSet(结果集) 对象表示符合 SQL 语句的所有行

我们进行查询操作:

String selectSql = "select * from employees";
PreparedStatement statement = connection.prepareStatement(sql);
ResultSet resultSet = statement.executeQuery();

ResultSet 中的数据是一行一行排列的,每行有多个字段,有一个记录指针,指针所指的数据行为当前数据行

我们只能操作当前的数据行,若我们想要获取某一行,就需要使用 ResultSet 的 next() 方法找到这一行,若我们想要得到 ResultSet 中的所有记录,就需要使用 while 循环

while (resultSet.next()) {

}

通过其提供的 get 方法来对行中的数据进行访问:

例如: 

        while (resultSet.next()) {
            int id = resultSet.getInt("id");
            String name = resultSet.getString("name");
            int managerId = resultSet.getInt("manager_id");
            System.out.println("id: " + id + " name: " + name + " manager_id: " + managerId);
        }

注意:结果集也需要释放

为了保证资源能够释放,我们可以将它们放到 finally

public class Test {
    public static void main(String[] args) {
        Connection connection = null;
        PreparedStatement statement = null;
        ResultSet resultSet = null;
        try {
            // 构造 DataSourse
            DataSource dataSource = new MysqlDataSource();
            ((MysqlDataSource) dataSource).setUrl("jdbc:mysql://127.0.0.1:3306/test?characterEncoding=utf8&useSSL=false");
            ((MysqlDataSource) dataSource).setUser("root");
            ((MysqlDataSource) dataSource).setPassword("phantom5170&");
            // 建立连接
            connection = dataSource.getConnection();
            // 构造 sql
            String sql = "select * from employees";
            statement = connection.prepareStatement(sql);
            resultSet = statement.executeQuery();
            // 查看结果集
            while (resultSet.next()) {
                int id = resultSet.getInt("id");
                String name = resultSet.getString("name");
                int managerId = resultSet.getInt("manager_id");
                System.out.println("id: " + id + " name: " + name + " manager_id: " + managerId);
            }

        } catch (SQLException e) {
            e.printStackTrace();
        } finally {
            // 执行完毕,释放资源
            if(resultSet != null) {
                try {
                    resultSet.close();
                } catch (SQLException e) {
                    e.printStackTrace();
                }
            }
            if(statement != null) {
                try {
                    statement.close();
                } catch (SQLException e) {
                    e.printStackTrace();
                }
            }
            if(connection != null) {
                try {
                    connection.close();
                } catch (SQLException e) {
                    e.printStackTrace();
                }
            }
        }
    }
}

 

使用总结

JDBC 使用步骤:

1. 创建数据库连接Connection

DriverManager 创建

DataSource 获取

2. 创建操作命令Statement

Statement

PreparedStatement

CallableStatement

3. 使用操作命令来执行SQL

executeQuery()

executeUpdate() 

4. 处理结果集 Result

while (resultSet.next()) {

        int id = resultSet.getInt("id");

        String name = resultSet.getString("name");

        ...

5. 释放资源 

 // 执行完毕,释放资源
            if(resultSet != null) {
                try {
                    resultSet.close();
                } catch (SQLException e) {
                    e.printStackTrace();
                }
            }
            if(statement != null) {
                try {
                    statement.close();
                } catch (SQLException e) {
                    e.printStackTrace();
                }
            }
            if(connection != null) {
                try {
                    connection.close();
                } catch (SQLException e) {
                    e.printStackTrace();
                }
            }

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/879589.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

20240920 每日AI必读资讯

阿里通义千问开源Qwen2.5系列模型:Qwen2-VL-72B媲美GPT-4 - Qwen2.5系列模型开源,包括通用语言模型和专业领域模型,提升知识获取、编程和数学能力。 - 模型支持长文本处理,生成最多8K tokens内容,对29种以上语言提供…

Java多线程面试精讲:源于技术书籍的深度解读

写在前面 ⭐️在无数次的复习巩固中,我逐渐意识到一个问题:面对同样的面试题目,不同的资料来源往往给出了五花八门的解释,这不仅增加了学习的难度,还容易导致概念上的混淆。特别是当这些信息来自不同博主的文章或是视…

SpringCloud系列之一---搭建高可用的Eureka注册中心

前言 本篇文章主要介绍的是SpringCloud相关知识、微服务架构以及搭建服务注册与发现的服务模块(Eureka)以及Eureka集群。 GitHub源码链接位于文章底部。 什么是SpringCloud Spring Cloud 是一系列框架的有序集合。 它利用 Spring Boot 的开发便利性巧妙地简化了分布式系统基础设…

ATGM331C-5T杭州中科微全星座定位授时模块电气参数

ATGM331C-5T 系列模块通过 UART 作为主要输出通道,按照 NMEA0183 的协议格式输出。 产品选型: 性能指标: 出色的定位导航功能,支持 BDS/GPS 卫星导航系统的单系统授时,以及任意组合的多系统联合定位,并支持…

【学习笔记】SSL/TLS证书安全机制之证书透明

1、概念 CT - Certificate Transparency,证书透明 2、Trying to Solve 如果意外的 CA 为我们的域名颁发证书,我们是不可见,这就是证书透明(CT)要解决的问题 3、How CT Works 任何CA机构颁发的所有证书的公共登记处&…

望繁信科技携流程智能解决方案亮相CNDS 2024新能源产业数智峰会

9月13日,CNDS 2024中国新能源产业数智峰会在北京圆满落幕。本次峰会以“走向数字新能源”为主题,汇聚了来自新能源领域的顶尖领袖、专家学者及知名企业代表,共同探讨数字化技术在新能源行业中的创新应用和发展趋势。上海望繁信科技有限公司&a…

网安标委发布敏感个人信息识别指南

9月14日全国网络安全标准化技术委员会秘书处发布《网络安全标准实践指南——敏感个人信息识别指南》 敏感个人信息识别规则: 一旦遭到泄露或者非法使用,容易导致自然人的人格尊严受到侵害、自然人的人身安全受到危害、自然人财产安全受到危害。 注意&am…

CISP备考题库(八)

CISP即“注册信息安全专业人员”,是面向信息安全企业、信息安全咨询服务机构、信息安全测评机构、政府机构、社会各组织、团体、大专院校以及企事业单位中负责信息系统建设、运行维护和管理工作的信息安全专业人员所颁发的专业资质证书。 更多CISP介绍:e…

【Git】常见命令(仅笔记)

文章目录 创建/初始化本地仓库添加本地仓库配置项提交文件查看仓库状态回退仓库查看日志分支删除文件暂存工作区代码远程仓库使用 .gitigore 文件让 git 不追踪一些文件标签 创建/初始化本地仓库 git init添加本地仓库配置项 git config -l #以列表形式显示配置项git config …

FTP、SFTP安装,整合Springboot教程

文章目录 前言一、FTP、SFTP是什么?1.FTP2.SFTP 二、安装FTP1.安装vsftp服务2.启动服务并设置开机自启动3.开放防火墙和SELinux4.创建用户和FTP目录4.修改vsftpd.conf文件5.启动FTP服务6.问题 二、安装SFTP1、 创建用户2、配置ssh和权限3、建立目录并赋予权限4、启动…

Elastic 的 OpenTelemetry PHP 发行版简介

作者:Pawel Filipczak 宣布 OpenTelemetry PHP 的 Elastic 发行版的第一个 alpha 版本。在本篇博文中了解使用 OpenTelemetry 来检测 PHP 应用程序是多么简单。 我们很高兴推出 OpenTelemetry PHP 的 Elastic Distribution 的第一个 alpha 版本。在这篇文章中&…

python植物大战僵尸项目源码【免费】

植物大战僵尸是一款经典的塔防游戏,玩家通过种植各种植物来抵御僵尸的进攻。 源码下载地址: 植物大战僵尸项目源码 提取码: 8muq

Ubuntu 22.04.5 LTS 发布下载 - 现代化的企业与开源 Linux

Ubuntu 22.04.5 LTS (Jammy Jellyfish) - 现代化的企业与开源 Linux Ubuntu 22.04.5 发布,配备 Linux 内核 6.8 请访问原文链接:https://sysin.org/blog/ubuntu-2204/,查看最新版。原创作品,转载请保留出处。 作者主页&#xf…

ICPC2024 邀请赛西安站 F L题解

F - XOR Game 题意 给定n,k ,k代表0的个数,现在有一个数x初始为0 接下来n个数,每一个数代表这个数字的个数 每次操作可以选择a数组中的一个数字并且可以选择是否将这个x异或上这个数字,然后把这个数字从a数组中删除,Alice先手,Alice想让答案尽可能大,Bob想让答案尽可能小,问…

腾讯音乐2024 Q2财报稳中有进,首席执行官梁柱(Ross Liang)强调平台创新

8 月 13 日,腾讯音乐娱乐集团(Tencent Music Entertainment Group,以下简称“TME”)发布 2024 年第二季度财报。本季度集团各项核心财务指标稳健增长,总收入达 71.6 亿元,调整后净利润 19.9 亿元&#xff0…

《Learning to Prompt for Vision-Language Models》CoOp论文中文校对版

系列论文研读目录 文章目录 系列论文研读目录摘要1 简介2 相关工作2.1视觉语言模型2.2 NLP中的提示学习 3 方法论3.1视觉语言预训练3.2上下文优化3.3讨论 4 实验4.1Few-Shot学习4.2领域泛化4.3进一步分析 5 结论、局限性和未来的工作 摘要 像CLIP这样的大型预训练视觉语言模型…

基于SpringBoot+Vue的篮球馆会员信息管理系统

作者:计算机学姐 开发技术:SpringBoot、SSM、Vue、MySQL、JSP、ElementUI、Python、小程序等,“文末源码”。 专栏推荐:前后端分离项目源码、SpringBoot项目源码、SSM项目源码 系统展示 【2025最新】基于JavaSpringBootVueMySQL的…

36.贪心算法3

1.坏了的计算器(medium) . - 力扣(LeetCode) 题目解析 算法原理 代码 class Solution {public int brokenCalc(int startValue, int target) {// 正难则反 贪⼼int ret 0;while (target > startValue) {if (target % 2 0…

深入理解中比较两个字符串差异的方法”或“高效比对字符串:diff-match-patch:c++实战指南

diff-match-patch 是一个强大的开源 JavaScript 库,由 Google 开发并维护,用于计算两个字符串之间的差异,并进行高效的匹配和补丁应用。这个库广泛应用于版本控制系统、协同编辑系统以及任何需要处理文本变化的场景。 GitHub地址:…

继承1 2024_9_18

1.继承的基本用法 当需要继承的时候,我们就在派生类的后面加上一个权限父类,这个权限可以是公有,保护和私有,后面就是继承的父类.此时,下面的stu这个派生类,也就可以使用Person里面的方法了. 2.继承基类成员访问方式的变化 当父类被继承到派生类的时候,此时会根据继承方式的不…