JAVA保姆式JDBC数据库免费教程之02-连接池技术

news2025/1/11 12:48:34

连接池

连接池概念

​ 概念:其实就是一个容器(集合),存放数据库连接的容器。

当系统初始化好后,容器被创建,容器中会申请一些连接对象,当用户来访问数据库时,从容器中获取连接对象,用户访问完之后,会将连接对象归还给容器。
在这里插入图片描述

缺点:用户每次请求都需要向数据库获得链接,而数据库创建连接通常需要消耗相对较大的资源,创建时间也较长。假设网站一天10万访问量,数据库服务器就需要创建10万次连接,极大的浪费数据库的资源,并且极易造成数据库服务器内存溢出。

连接池技术的核心思想是:连接复用,通过建立一个数据库连接池以及一套连接使用、分配、管理策略,使得该连接池中的连接可以得到高效、安全的复用,避免了数据库连接频繁建立、关闭的开销。

在这里插入图片描述

连接池好处

	1. 节约资源	
	2. 减少连接db的次数	
	3. 用户访问高效

连接池实现

1. 标准接口:DataSource   javax.sql包下的
1. 方法:
	* 获取连接:getConnection()
	* 归还连接:Connection.close()。如果连接对象Connection是从连接池中获取的,那么调用Connection.close()方法,则不会再关闭连接了。而是归还连接。

厂商实现

一般我们不去实现它,有数据库厂商来实现
    DBCP 
	1. C3P0:数据库连接池技术
	2. Druid:数据库连接池实现技术,由阿里巴巴提供的

c3p0数据库连接池

实现步骤

1. 导入jar包  c3p0-0.9.5.2.jar 
新建web项目 C3p0Test
使用c3p0的连接池,导入jar包
导入 mysql驱动包

2. 定义配置文件:
名称: c3p0.properties 或者 c3p0-config.xml  
路径:直接将文件放在src目录下即可。

3. 创建核心对象 数据库连接池对象 ComboPooledDataSource

4. 获取连接: getConnection
代码:
//1.创建数据库连接池对象
DataSource ds  = new ComboPooledDataSource();
//2. 获取连接对象
Connection conn = ds.getConnection();

工具类的抽取

package cn.yanqi.utils;
import java.sql.Connection;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.sql.Statement;
import com.mchange.v2.c3p0.ComboPooledDataSource;

// 你抽出的代码要有一定意义,并且要有通用性,任何一个关于这个工具类都 可以去使
public class JdbcUtils {
    //默认加载src 下面c3p0-config.xml加载
    private static ComboPooledDataSource ds = new ComboPooledDataSource();

    //创建连接
    public static Connection getConnection() throws SQLException{
        return ds.getConnection();
    }
}

配置文件

c3p0-config.xml 注意一定放在src下

<?xml version="1.0" encoding="UTF-8"?>
<c3p0-config>
    <!-- 我们希望在配置文件中,出现链接的参数信息 -->
    <default-config>
        <!-- name 属性定义 链接参数的key 标签的内容 代表值-->
        <property name="driverClass">com.mysql.jdbc.Driver</property>
        <property name="jdbcUrl">jdbc:mysql:///day03</property>
        <property name="user">root</property>
        <property name="password">root</property>

		<!-- 连接池参数 -->
		<property name="initialPoolSize">5</property>
        <property name="maxPoolSize">10</property>
        <property name="checkoutTimeout">3000</property>

    </default-config>
</c3p0-config>

测试

@Test
public void test1() throws SQLException {

    //封装的c3p0相关代码
    Connection conn = JdbcUtil.getConnection();
    Statement stmt = conn.createStatement();
    int i = stmt.executeUpdate("delete from user where id = 29");
    System.out.println(i);

}

Druid数据库连接池

Druid数据库连接池实现技术 (德鲁伊),由阿里巴巴提供的

实现步骤

一、步骤:
	1. 导入jar包 druid-1.0.9.jar
	2. 定义配置文件:
		是xx.properties形式的
		可以叫任意名称,可以放在任意目录下, 一般都放在src
	3. 加载配置文件。Properties
	4. 获取数据库连接池对象:通过工厂来来获取  DruidDataSourceFactory
	5. 获取连接:getConnection
	
代码实现:
		//3.加载配置文件
	     Properties pro = new Properties();
	     InputStream is = 		       DruidDemo.class.getClassLoader().getResourceAsStream("druid.properties");
	     pro.load(is);
	     //4.获取连接池对象
	     DataSource ds = DruidDataSourceFactory.createDataSource(pro);
	     //5.获取连接
	     Connection conn = ds.getConnection();
二、定义工具类
		1. 定义一个类 JDBCUtils
		2. 提供静态代码块加载配置文件,初始化连接池对象
		3. 提供方法
			1. 获取连接方法:通过数据库连接池获取连接
			2. 释放资源
			3. 获取连接池的方法

配置文件

在src中创建 druid.properties

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

工具类的抽取

import java.io.IOException;
import java.sql.ResultSet;
import java.sql.SQLException;
/**
 * Druid连接池的工具类
 */
public class JDBCUtils {

    //1.定义成员变量 DataSource
    private static DataSource ds ;

    static{
        try {
            //1.加载配置文件
            Properties pro = new Properties();
            pro.load(JDBCUtils.class.getClassLoader().getResourceAsStream("druid.properties"));
            //2.获取DataSource
            ds = DruidDataSourceFactory.createDataSource(pro);
        } catch (IOException e) {
            e.printStackTrace();
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    /**
     * 获取连接
     */
    public static Connection getConnection() throws SQLException {
        return ds.getConnection();
    }
    
    /**
     * 获取DataSource对象 (用于jdbcTemplate获取)
     */
    public static DataSource getDataSource(){
        return ds;
    }
}

测试

public class DruidTest {

    @Test
    public void test(){

        Connection conn = DruidUtil.getConnection();
        try {
            Statement stmt = conn.createStatement();
            int i = stmt.executeUpdate("INSERT INTO `test`.`t_user`(`id`, `username`, `passworld`, `age`) VALUES (null, '江一燕', '520', 18)");
            System.out.println(i);
        } catch (SQLException e) {
            e.printStackTrace();
        }
    }
}

JDBCTemplate模版

概述

Spring JDBC Spring框架对JDBC的简单封装。提供了一个JDBCTemplate对象简化JDBC的开发(后面我们专门讲spring框架

实现步骤

1. 导入jar包 4 + 1

2. 创建JdbcTemplate对象。依赖于数据源DataSource
	JdbcTemplate template = new JdbcTemplate(ds);

3. 调用JdbcTemplate的方法来完成CRUD的操作
	update():执行DML语句。增、删、改语句
	queryForMap():查询结果将结果集封装为map集合,将列名作为key,将值作为value 将这条记录封装为一个map集合
			注意:这个方法查询的结果集长度只能是1
	queryForList():查询结果将结果集封装为list集合
			注意:将每一条记录封装为一个Map集合,再将Map集合装载到List集合中
	query():查询结果,将结果封装为JavaBean对象
			query的参数:RowMapper
			一般我们使用BeanPropertyRowMapper实现类。可以完成数据到JavaBean的自动封装
			new BeanPropertyRowMapper<类型>(类型.class)
			queryForObject:查询结果,将结果封装为对象
			一般用于聚合函数的查询

快速入门

/**
 * JdbcTemplate入门
 */
public class JdbcTemplateDemo1 {
    public static void main(String[] args) {
        //1.导入jar包
        //2.创建JDBCTemplate对象
        JdbcTemplate template = new JdbcTemplate(JDBCUtils.getDataSource());
        //3.调用方法
        String sql = "update account set balance = 5000 where id = ?";
        int count = template.update(sql, 3);//insert update delete	
        System.out.println(count);
        //自已关闭,自已归还
    }
}

CRUD的测试

代码实现

数据库准备

CREATE TABLE `emp` (
  `id` int(11) NOT NULL AUTO_INCREMENT,
  `username` varchar(25) DEFAULT NULL,
  `job` varchar(100) DEFAULT NULL,
  `addr` varchar(25) DEFAULT NULL,
  PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8;

INSERT INTO `test`.`emp`(`id`, `username`, `job`, `addr`) VALUES (1, 'tom', '开发', '上海');
INSERT INTO `test`.`emp`(`id`, `username`, `job`, `addr`) VALUES (2, 'mary', '测试', '郑州');

实体类

// 注意事项:java属性名   要与   数据库的字段名保持一致,不然无法封装
public class Emp {
    private Integer id;
    private String username;
    private String job;
    private String addr;
    //提供有参无对构造方法
    //提供get  set 方法
    //提供toString方法
}

测试类

public class Test1 {

    //获取数据源
    private DataSource dataSource = JDBCUtil.getDataSource();
    //使用jdbcTemplate 技术,简化jdbc操作
    private JdbcTemplate JdbcTemplate = new JdbcTemplate(dataSource);

    //修改
    @Test
    public void testUpdateById(){
        String sql ="update emp set job = '财务' where id = ? ";
        int i = this.JdbcTemplate.update(sql, 2);
        System.out.println(i);
    }

    //添加
    @Test
    public void testAdd(){
        String sql ="insert into emp (id , username , job ,addr) values(?,?,?,?)";
        int i = JdbcTemplate.update(sql, null, "yiyan", "小秘", "郑州");
        System.out.println(i);
    }
    //删除
    @Test
    public void testDeleteById(){
        String sql ="delete from emp where id = ?";
        int i = JdbcTemplate.update(sql, 1);
        System.out.println(i);
    }

    //查询数据返回list
    @Test
    public void testQueryById(){
        String sql ="select * from emp where id = ?";
        //返回的是list
        List<Emp> emps = this.JdbcTemplate.query(sql, new BeanPropertyRowMapper<>(Emp.class), 2);
        for(Emp e : emps){
            System.out.println(e);
        }
    }

    //查询条数
    @Test
    public void testForObject(){
        String sql = "select count(1) from emp";
        Long i = this.JdbcTemplate.queryForObject(sql, Long.class);
        System.out.println(i);
    }

    //查询数据返回map
    @Test
    public void testQuery2(){
        String sql = "select * from emp where id = ?";
        //返回map
        Map<String, Object> map = this.JdbcTemplate.queryForMap(sql,2);
        //map遍历
        Set<String> set = map.keySet();
        for(String key : set){
            Object o = map.get(key);
            System.out.println(key+" : "+o);
        }
    }
    
     /**
     * 查询数据返回对象中
     * 注意事项: 返回的单个对象必须要有结果,这个结果只能接受一条
     *           
     */
    public  Emp queryTest3(){
        Emp emp = null;
        try {
            String sql = "select * from emp where id = ?";
            //如果这里查到数据,就直接返回
            return this.template.queryForObject(sql, new BeanPropertyRowMapper<>(Emp.class),9);
        } catch (DataAccessException e) {
            //没有查询到,方式一
            // throw new RuntimeException("没有查询到数据!");
            //没有查询到,方式二
            return null;
        }
    }
	//测试 查询数据返回对象中
    public static void main(String[] args) {
        JdbcTemplateDemo jt = new JdbcTemplateDemo();
        Emp emp = jt.queryTest3();
        System.out.println(emp);
    }
    
    
}

<>(Emp.class),9);
} catch (DataAccessException e) {
//没有查询到,方式一
// throw new RuntimeException(“没有查询到数据!”);
//没有查询到,方式二
return null;
}
}
//测试 查询数据返回对象中
public static void main(String[] args) {
JdbcTemplateDemo jt = new JdbcTemplateDemo();
Emp emp = jt.queryTest3();
System.out.println(emp);
}

}


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

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

相关文章

【MySQL】MySQL 架构

一、MySQL 架构 C/S 架构&#xff0c;即客户端/服务器架构。服务器程序直接和我们存储的数据打交道&#xff0c;多个客户端连接这个服务器程序。客户端发送请求&#xff0c;服务器响应请求。 MySQL 数据库实例 &#xff1a;即 MySQL 服务器的进程 &#xff08;我们使用任务管理…

Vue组件间通信的四种方式(函数回调,自定义事件,事件总线,消息订阅与发布)

目录 概述 props配置项-回调函数实现 自定义事件实现 事件总线实现 消息订阅与发布实现 概述 在组件化编程中&#xff0c;组件间的通信是重要的&#xff0c;我们可以有四种方式实现组件间的通信。 分别是&#xff1a;函数回调&#xff0c;自定义事件&#xff0c;事件总…

可调恒流驱动LED电路分析

https://www.icxbk.com/article/detail?aid884 常规使用的pwm调亮度不仅会导致频闪&#xff0c;而且在长时间使用的时候&#xff0c;有损坏led的风险&#xff0c;所以这次设计了一个恒流调亮度电路&#xff0c;其电路图如下所示 电路原理的解读&#xff1a; 左侧的电位计起着…

【JavaScript】js实现深拷贝的方法

前言 在js中我们想要实现深拷贝&#xff0c;首先要了解深浅拷贝的区别。 浅拷贝&#xff1a;只是拷贝数据的内存地址&#xff0c;而不是在内存中重新创建一个一模一样的对象&#xff08;数组&#xff09; 深拷贝&#xff1a;在内存中开辟一个新的存储空间&#xff0c;完完全全…

Java语言常用哪些运算符?

之前有个大家讨论过java的数据类型&#xff0c;总体来说类型和其他几种语言也相差无几&#xff0c;我为什么会这样说&#xff1f;我们应该都要知道Python可还有个复数类型。 这里主要给大家讲解Java运算符的分类和使用。 一、运算符分类 说到运算符&#xff0c;我们可以先了…

硬件系统工程师宝典(9)-----如何正确使用去耦电容

各位同学大家好&#xff0c;欢迎继续做客电子工程学习圈&#xff0c;今天我们继续来讲这本书&#xff0c;硬件系统工程师宝典。上篇我们说到在电源完整性分析时&#xff0c;明确噪声来源可以有效的避免、解决噪声问题。今天我们来看看电源完整性分析中重要的一环&#xff0c;去…

【自动化测试】web自动化测试验证码如何测?如何处理验证码问题?解决方案......

目录&#xff1a;导读前言一、Python编程入门到精通二、接口自动化项目实战三、Web自动化项目实战四、App自动化项目实战五、一线大厂简历六、测试开发DevOps体系七、常用自动化测试工具八、JMeter性能测试九、总结&#xff08;尾部小惊喜&#xff09;前言 在对安全性有要求的…

线程池ThreadPoolExecutor源码剖析

一、Java构建线程的方式 继承Thread &#xff08;也实现了Runnable&#xff09; 实现Runnable 实现Callable &#xff08;与Runnable区别…&#xff09; 线程池方式 &#xff08;Java提供了构建线程池的方式&#xff09;[可以实现Runnable 和 Callable 功能] Java提供了Exe…

使用Vue3实现一个可复制的表格

前言 表格是前端非常常用的一个控件&#xff0c;但是每次都使用v-for指令手动绘制tr/th/td这些元素是非常麻烦的。同时&#xff0c;基础的 table 样式通常也是不满足需求的&#xff0c;因此一个好的表格封装就显得比较重要了。 最基础的表格封装 最基础基础的表格封装所要做…

【并发编程十七】c++实现一个线程池

【并发编程十七】c实现一个线程池一、线程池原理二、实现重点三、个人理解四、实验简介&#xff1a; 大多数系统上&#xff0c;若因某些任务可以与其他任务并行处理&#xff0c;就分别给他们配备专属的线程&#xff0c;则这种做法不切实际。但是只要有可能&#xff0c;我们还是…

C语言进阶——动态内存管理(上)

&#x1f307;个人主页&#xff1a;_麦麦_ &#x1f4da;今日名言&#xff1a;“你若爱&#xff0c;生活哪里都可爱。你若恨&#xff0c;生活哪里都可恨。你若感恩&#xff0c;处处可感恩。你若成长&#xff0c;事事可成长。不是世界选择了你&#xff0c;是你选择了这个世界。既…

mdio协议

1. 简介 MDIO接口中有特定的术语定义总线上的各种设备&#xff0c;驱动MDIO总线的设备被定义为站管理实体&#xff08;STA&#xff09;&#xff0c;而被MDC管理的目标设备称为可被MDIO管理的设备&#xff08;MMD&#xff09;。 STA初始化MDIO所有的通信&#xff0c;同时负责驱动…

【数据结构与算法】哈希表1:字母异位词 两数交集 快乐数 两数之和

文章目录今日任务1.哈希表理论基础&#xff08;1&#xff09;哈希表&#xff08;2&#xff09;哈希函数&#xff08;3&#xff09;哈希碰撞&#xff08;4&#xff09;链地址法&#xff08;拉链法&#xff09;&#xff08;5&#xff09;线性探测法&#xff08;6&#xff09;常见…

Python采集双色球数据,做数据分析,让我自己实现自己的富豪梦

来唠点嗑&#xff1f; 咳咳&#xff0c;最近是咋的了&#xff0c;某站掀起了一股双色球热潮&#xff1f;一般我自己的账号上&#xff0c;是很少看到关于python这些内容的&#xff0c;都是小姐姐和热梗&#xff0c;或者其他搞笑视频 由于&#x1f4b4;的吸引力…手不自觉的就点…

《系统架构设计》-03-软件结构体系和架构风格

文章目录1. 软件结构体系1.1 抽象&#xff08;Abstract&#xff09;1.1.1 抽象的应用1.1.2 不同层次的抽象1.2 组件&#xff08;Component&#xff09;1.2.1 定义1.2.2 切入点1.3 组织过程资产&#xff08;Organizational Process Assets&#xff09;1.3.1 定义1.3.2 作用1.4 体…

springboot整合Chat Generative Pre-trained Transformer

什么是Chat Generative Pre-trained Transformer Chat Generative Pre-trained Transformer&#xff0c;是以人工智能驱动的聊天机器人程序 &#xff0c;已经更新多个版本&#xff0c;很多大厂也都在接入其API。 整合难度 难度一颗星&#xff0c;基本上就是给官方API发请求&am…

特征工程:特征构造以及时间序列特征构造

数据和特征决定了机器学习的上限&#xff0c;而模型和算法只是逼近这个上限而已。由此可见&#xff0c;特征工程在机器学习中占有相当重要的地位。在实际应用当中&#xff0c;可以说特征工程是机器学习成功的关键。 那特征工程是什么&#xff1f; 特征工程是利用数据领域的相关…

UI自动化测试之设计框架

目的 相信做过测试的同学都听说过自动化测试&#xff0c;而UI自动化无论何时对测试来说都是比较吸引人的存在。 相较于接口自动化来说它可以最大程度的模拟真实用户的日常操作与特定业务场景的模拟&#xff0c;那么存在即合理&#xff0c;自动化UI测试自然也是广大测试同学职…

身为大学生,你不会还不知道有这些学生福利吧!!!!

本文介绍的是利用学生身份可以享受到的相关学生优惠权益&#xff0c;但也希望各位享受权利的同时不要忘记自己的义务&#xff0c;不要售卖、转手自己的学生优惠资格&#xff0c;使得其他同学无法受益。 前言 高考已经过去&#xff0c;我们也将迎来不同于以往的大学生活&#x…

磁盘结构

一.盘片 盘片是一个圆形坚硬的表面&#xff0c;通过引入磁性变化来永久存储数据&#xff0c;这些盘片通常由一些硬质材料&#xff08;如铝&#xff09;制成&#xff0c;然后涂上薄薄的磁性层&#xff0c;即使驱动器断电&#xff0c;驱动器也能持久存储数据位。每个盘片有两面&a…