深入浅出 JDBC 与数据库连接池

news2025/5/11 23:35:35

在Java开发中,与数据库进行交互是几乎每个项目都离不开的功能。JDBC(Java DataBase Connectivity)作为Java操作数据库的标准规范,为开发者提供了底层的数据库访问支持。而数据库连接池则是提高数据库操作效率和性能的重要工具。本文将结合实际代码示例,带你深入浅出地了解JDBC和数据库连接池的知识与应用。

一、JDBC 基础入门

(一)JDBC 简介

JDBC 是一组 Java API,它允许 Java 程序与各种关系型数据库进行交互。通过 JDBC,我们可以连接到数据库、执行 SQL 语句、处理查询结果等操作。JDBC 的核心类和接口主要集中在 java.sqljavax.sql 包中。

(二)JDBC 开发步骤

  1. 注册驱动 :告知程序要使用的数据库驱动。

    • 早期做法为使用 DriverManager.registerDriver(new DriverImpl()),但这种方式可能会导致驱动被注册两次。

    • 推荐做法是使用 Class.forName("数据库驱动类名"),例如 Class.forName("com.mysql.jdbc.Driver"),这种方式会自动加载驱动。

  2. 获取数据库连接 :通过 DriverManager.getConnection(url, username, password) 方法获取连接。其中,url 指定数据库的连接地址,如 jdbc:mysql://localhost:3306/数据库名

  3. 创建 Statement 对象 :使用 Connection.createStatement() 方法创建,用于执行 SQL 语句。

  4. 执行 SQL 语句

    • 查询操作:调用 Statement.executeQuery(sql) 方法执行查询,返回 ResultSet 对象,其中包含查询结果。

    • 增删改操作:调用 Statement.executeUpdate(sql) 方法执行,返回受影响的行数。

  5. 处理结果集(针对查询操作) :通过 ResultSet 对象提供的方法(如 next()getInt()getString() 等)遍历查询结果并获取数据。

  6. 释放资源 :按照 resultSet → statement → connection 的顺序依次关闭,避免资源泄漏。通常在 finally 块中进行关闭操作。

(三)代码示例:JDBC 查询操作

import java.sql.*;

public class JdbcDemo {
    public static void main(String[] args) {
        Connection connection = null;
        Statement statement = null;
        ResultSet resultSet = null;
        try {
            // 注册驱动
            Class.forName("com.mysql.jdbc.Driver");
            // 获取连接
            connection = DriverManager.getConnection("jdbc:mysql:///spring_db", "root", "2020");
            // 创建 Statement
            statement = connection.createStatement();
            // 执行查询
            String sql = "select * from account";
            resultSet = statement.executeQuery(sql);
            // 处理结果集
            while (resultSet.next()) {
                int id = resultSet.getInt("id");
                String name = resultSet.getString("name");
                double money = resultSet.getDouble("money");
                System.out.println("id: " + id + ", name: " + name + ", money: " + money);
            }
        } catch (Exception e) {
            e.printStackTrace();
        } finally {
            // 释放资源
            try {
                if (resultSet != null)
                    resultSet.close();
                if (statement != null)
                    statement.close();
                if (connection != null)
                    connection.close();
            } catch (SQLException e) {
                e.printStackTrace();
            }
        }
    }
}

二、JDBC 工具类与配置文件

(一)工具类的作用

为了简化 JDBC 操作代码,提高代码的可维护性和复用性,通常会编写 JDBC 工具类。工具类可以封装注册驱动、获取连接、关闭资源等公共操作。

(二)基于配置文件的工具类

将数据库连接的相关信息(如驱动类名、url、用户名、密码等)配置在 properties 文件中,工具类通过读取配置文件来获取这些信息,使代码更具灵活性,便于后期修改。

1. 配置文件(db.properties)
driverclass=com.mysql.jdbc.Driver
url=jdbc:mysql:///spring_db
username=root
password=2020
2. 工具类代码(JdbcUtils2.java)
import java.io.*;
import java.sql.*;
import java.util.Properties;

public class JdbcUtils2 {
    private static final String driverclass;
    private static final String url;
    private static final String username;
    private static final String password;

    static {
        Properties properties = new Properties();
        InputStream inputStream = JdbcUtils2.class.getResourceAsStream("/db.properties");
        try {
            properties.load(inputStream);
        } catch (IOException e) {
            e.printStackTrace();
        }
        driverclass = properties.getProperty("driverclass");
        url = properties.getProperty("url");
        username = properties.getProperty("username");
        password = properties.getProperty("password");
    }

    public static void createDriver() {
        try {
            Class.forName(driverclass);
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        }
    }

    public static Connection getConnection() {
        Connection connection = null;
        createDriver();
        try {
            connection = DriverManager.getConnection(url, username, password);
        } catch (SQLException e) {
            e.printStackTrace();
        }
        return connection;
    }

    public static void close(Connection connection, ResultSet resultSet, Statement statement) throws SQLException {
        if (resultSet != null)
            resultSet.close();
        if (statement != null)
            statement.close();
        if (connection != null)
            connection.close();
    }

    public static void close(Connection connection, Statement statement) throws SQLException {
        if (statement != null)
            statement.close();
        if (connection != null)
            connection.close();
    }
}

三、JDBC 事务管理

(一)事务的概念

事务是数据库中一系列操作的集合,这些操作要么全部成功,要么全部失败回滚。事务具有 ACID 特性:

  • 原子性(Atomicity) :事务中的操作不可分割,要么全部执行,要么全部不执行。

  • 一致性(Consistency) :事务执行前后,数据库始终处于一致的状态,符合业务规则和完整性约束。

  • 隔离性(Isolation) :多个事务并发执行时,相互之间不会产生干扰,一个事务的执行不能被其他事务干扰。

  • 持久性(Durability) :一旦事务提交,其对数据库的修改将永久保存,即使系统发生故障也不会丢失。

(二)JDBC 中操作事务

在 JDBC 中,通过 Connection 对象来管理事务。

  1. 开启事务 :调用 connection.setAutoCommit(false) 方法,将自动提交模式设置为 false,从而开启事务。

  2. 提交事务 :调用 connection.commit() 方法提交事务,使事务中的所有操作永久生效。

  3. 回滚事务 :调用 connection.rollback() 方法回滚事务,撤销事务中的所有操作,使数据库回到事务开始前的状态。

(三)事务的隔离级别

不同的事务隔离级别可以解决不同的并发问题:

  • 读未提交(Read Uncommitted) :最低的隔离级别,事务可以读取其他事务未提交的数据,可能导致脏读、不可重复读和幻读问题。

  • 读已提交(Read Committed) :事务只能读取其他事务已经提交的数据,避免了脏读,但可能仍然存在不可重复读和幻读问题。

  • 可重复读(Repeatable Read) :保证事务在多次读取同一数据时,结果一致,避免了脏读和不可重复读,但可能出现幻读问题。这是 MySQL 的默认隔离级别。

  • 串行化(Serializable) :最高的隔离级别,通过强制事务串行执行,避免了脏读、不可重复读和幻读问题,但性能开销较大。

在 JDBC 中,可以通过 connection.setTransactionIsolation(level) 方法设置事务的隔离级别,其中 level 是上述隔离级别的对应常量。

四、数据库连接池

(一)连接池的概念

数据库连接池是应用程序与数据库之间的一组数据库连接的缓存。它预先创建并维护一定数量的数据库连接,当应用程序需要执行数据库操作时,可以从连接池中获取空闲连接,使用完毕后将连接归还到连接池中,而不是直接关闭连接。这样可以避免频繁地创建和销毁数据库连接所带来的性能开销,提高数据库操作的效率。

(二)常见的开源连接池

  1. DBCP(Database Connection Pool) :Apache 提供的开源连接池,配置简单,但在高并发场景下性能表现一般。

  2. C3P0 :另一个开源连接池,提供了自动回收空闲连接等功能,配置较为灵活。

  3. Druid :阿里巴巴开源的数据库连接池,功能强大、性能优秀,还提供了监控和分析数据库访问性能的能力。

(三)使用 Druid 连接池

1. 配置文件(druid.properties)
driverClassName=com.mysql.jdbc.Driver
url=jdbc:mysql:///spring_db
username=root
password=2020
initialSize=5
maxActive=10
maxWait=3000
maxIdle=6
minIdle=3
2. 工具类代码(JdbUtils3.java)
import com.alibaba.druid.pool.DruidDataSourceFactory;
import java.io.*;
import java.sql.*;
import java.util.Properties;

public class JdbUtils3 {
    private static DataSource dataSource;

    static {
        Properties properties = new Properties();
        InputStream inputStream = JdbcUtils2.class.getResourceAsStream("/druid.properties");
        try {
            properties.load(inputStream);
            dataSource = DruidDataSourceFactory.createDataSource(properties);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    public static Connection getConnection() {
        Connection connection = null;
        try {
            connection = dataSource.getConnection();
        } catch (SQLException e) {
            e.printStackTrace();
        }
        return connection;
    }

    public static void close(Connection connection, ResultSet resultSet, Statement statement) throws SQLException {
        if (resultSet != null)
            resultSet.close();
        if (statement != null)
            statement.close();
        if (connection != null)
            connection.close();
    }

    public static void close(Connection connection, Statement statement) throws SQLException {
        if (statement != null)
            statement.close();
        if (connection != null)
            connection.close();
    }
}

(四)连接池的优势

  1. 性能提升 :减少了数据库连接的创建和销毁时间,提高了数据库操作的响应速度。

  2. 资源复用 :连接池中的连接可以被多个应用程序或线程复用,避免了资源的浪费。

  3. 管理方便 :连接池提供了对连接的统一管理,可以方便地设置最大连接数、最小连接数、超时时间等参数,以适应不同的应用场景和负载需求。

总的来说,JDBC 是 Java 操作数据库的基础,而数据库连接池则是提升数据库操作性能的关键。在实际开发中,我们通常会结合使用 JDBC 和连接池,以实现高效、稳定、可靠的数据库访问。通过合理地配置和使用连接池,可以充分发挥其优势,提高系统的整体性能和可扩展性。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

16前端项目----交易页

交易 交易页Trade修改默认地址商品清单reduce计算总数和总价应用 统一引入接口提交订单 交易页Trade 在computed中mapState映射出addressInfo和orderInfo&#xff0c;然后v-for渲染到组件当中 修改默认地址 <div class"address clearFix" v-for"address in …

2003-2020年高铁线路信息数据

2003-2020年高铁线路信息数据 1、时间&#xff1a;2003-2020年 2、来源&#xff1a;Chinese High-speed Rail and Airline Database&#xff0c;CRAD 3、指标&#xff1a;高铁线路名称、起点名、终点名、开通时间、线路长度(km)、设计速度(km/h&#xff09;、沿途主要车站 …

MySQL COUNT(*) 查询优化详解!

目录 前言1. COUNT(*) 为什么慢&#xff1f;—— InnoDB 的“计数烦恼” &#x1f914;2. MySQL 执行 COUNT(*) 的方式 (InnoDB)3. COUNT(*) 优化策略&#xff1a;快&#xff01;准&#xff01;狠&#xff01;策略一&#xff1a;利用索引优化带 WHERE 子句的 COUNT(*) (最常见且…

nginx配置协议

1. 7层协议 OSI&#xff08;Open System Interconnection&#xff09;是一个开放性的通行系统互连参考模型&#xff0c;他是一个定义的非常好的协议规范&#xff0c;共包含七层协议。直接上图&#xff0c;这样更直观些&#xff1a; 1.1 协议配置 1.1.1 7层配置 这里我们举例…

UE5 PCG学习笔记

https://www.bilibili.com/video/BV1onUdY2Ei3/?spm_id_from333.337.search-card.all.click&vd_source707ec8983cc32e6e065d5496a7f79ee6 一、安装PCG 插件里选择以下进行安装 移动目录后&#xff0c;可以使用 Update Redirector References&#xff0c;更新下&#xff0…

《用MATLAB玩转游戏开发》打砖块:向量反射与实时物理模拟MATLAB教程

《用MATLAB玩转游戏开发&#xff1a;从零开始打造你的数字乐园》基础篇&#xff08;2D图形交互&#xff09;-《打砖块&#xff1a;向量反射与实时物理模拟》MATLAB教程 &#x1f3ae; 文章目录 《用MATLAB玩转游戏开发&#xff1a;从零开始打造你的数字乐园》基础篇&#xff08…

vue配置代理解决前端跨域的问题

文章目录 一、概述二、报错现象三、通过配置代理来解决修改request.js中的baseURL为/api在vite.config.js中增加代理配置 四、参考资料 一、概述 跨域是指由于浏览器的同源策略限制&#xff0c;向不同源(不同协议、不同域名、不同端口)发送ajax请求会失败 二、报错现象 三、…

java+vert.x实现内网穿透jrp-nat

用java vert.x开发一个内网穿透工具 内网穿透概述技术原理常见内网穿透工具用java vert.x开发内网穿透工具 jrp-nat为什么用java开发内网穿透工具&#xff1f;jrp-nat功能实现图解jrp-nat内网穿透工具介绍jrp-nat内网穿透工具特点jrp-nat软件架构jrp-nat安装教程jrp-nat程序下载…

【程序员AI入门:应用开发】8.LangChain的核心抽象

一、 LangChain 的三大核心抽象 1. ChatModel&#xff08;聊天模型&#xff09; 核心作用&#xff1a;与大模型&#xff08;如 GPT-4、Claude&#xff09;交互的入口&#xff0c;负责处理输入并生成输出。关键功能&#xff1a; 支持同步调用&#xff08;model.invoke&#xf…

每天五分钟机器学习:KTT条件

本文重点 在前面的课程中,我们学习了拉格朗日乘数法求解等式约束下函数极值,如果约束不是等式而是不等式呢?此时就需要KTT条件出手了,KTT条件是拉格朗日乘数法的推广。KTT条件不仅统一了等式约束与不等式约束的优化问题求解范式,KTT条件给出了这类问题取得极值的一阶必要…

Facebook的元宇宙新次元:社交互动如何改变?

科技的浪潮正将我们推向一个全新的时代——元宇宙时代。Facebook&#xff0c;这个全球最大的社交网络平台&#xff0c;已经宣布将公司名称更改为 Meta&#xff0c;全面拥抱元宇宙概念。那么&#xff0c;元宇宙究竟是什么&#xff1f;它将如何改变我们的社交互动方式呢&#xff…

概统期末复习--速成

随机事件及其概率 加法公式 推三个的时候ABC&#xff0c;夹逼准则 减法准则 除法公式 相互独立定义 两种分析 两个解法 古典概型求概率&#xff08;排列组合&#xff09; 分步相乘、分类相加 全概率公式和贝叶斯公式 两阶段问题 第一个小概率*A在小概率的概率。。。累计 …

n8n系列(1)初识n8n:工作流自动化平台概述

1. 引言 随着各类自动化工具的涌现,n8n作为一款开源的工作流自动化平台,凭借其灵活性、可扩展性和强大的集成能力,正在获得越来越多技术团队的青睐。 本文作为n8n系列的开篇,将带您全面了解这个强大的自动化平台,探索其起源、特性以及与其他工具的差异,帮助您判断n8n是否…

QT6 源(82):阅读与注释日历类型 QCalendar,本类并未完结,儒略历,格里高利历原来就是公历,

&#xff08;1&#xff09;本代码来自于头文件 qcalendar . h &#xff1a; #ifndef QCALENDAR_H #define QCALENDAR_H#include <limits>#include <QtCore/qglobal.h> #include <QtCore/qlocale.h> #include <QtCore/qstring.h> #include <QtCore/…

CVE体系若消亡将如何影响网络安全防御格局

CVE体系的核心价值与当前危机 由MITRE运营的通用漏洞披露&#xff08;CVE&#xff09;项目的重要性不容低估。25年来&#xff0c;它始终是网络安全专业人员理解和缓解安全漏洞的基准参照系。通过提供标准化的漏洞命名与分类方法&#xff0c;这套体系为防御者建立了理解、优先级…

OpenKylin安装Elastic Search8

一、环境准备 Java安装 安装过程此处不做赘述&#xff0c;使用以下命令检查是否安装成功。 java -version 注意&#xff1a;Elasticsearch 自 7.0 版本起内置了 OpenJDK&#xff0c;无需单独安装。但如需自定义 JDK&#xff0c;可设置 JAVA_HOME。 二、安装Elasticsearch …

【ARM AMBA AHB 入门 3 -- AHB 总线介绍】

请阅读【ARM AMBA 总线 文章专栏导读】 文章目录 AHB Bus 简介AHB Bus 构成AHB BUS 工作机制AHB 传输阶段 AHB InterfacesAHB仲裁信号 AHB 数据访问零等待传输(no waitstatetransfer)等待传输(transfers with wait states)多重传送(multipletransfer)--Pipeline AHB 控制信号 A…

多模态大模型中的视觉分词器(Tokenizer)前沿研究介绍

文章目录 引言MAETok背景方法介绍高斯混合模型&#xff08;GMM&#xff09;分析模型架构 实验分析总结 FlexTok背景方法介绍模型架构 实验分析总结 Emu3背景方法介绍模型架构训练细节 实验分析总结 InternVL2.5背景方法介绍模型架构 实验分析总结 LLAVA-MINI背景方法介绍出发点…

sqli-labs靶场第二关——数字型

一&#xff1a;查找注入类型&#xff1a; 输入 ?id1--与第一关的差别&#xff1a;报错; 说明不是字符型 渐进测试&#xff1a;?id1--&#xff0c;结果正常&#xff0c;说明是数字型 二&#xff1a;判断列数和回显位 ?id1 order by 3-- 正常&#xff0c; 说明有三列&am…

[模型选择与调优]机器学习-part4

七 模型选择与调优 1 交叉验证 (1) 保留交叉验证HoldOut HoldOut Cross-validation&#xff08;Train-Test Split&#xff09; 在这种交叉验证技术中&#xff0c;整个数据集被随机地划分为训练集和验证集。根据经验法则&#xff0c;整个数据集的近70%被用作训练集&#xff…