搞懂TomcatJdbc之连接池初始化

前言

TomcatJdbc数据库连接池最初是设计来用于替代Apache Commons DBCP,但现在TomcatJdbc数据库连接池更多是与DruidHikariCP数据库连接池进行比较,尽管扩展性不及Druid,性能不及HikariCP,但TomcatJdbc数据库连接池依旧有较为广泛的使用,这主要得益于其简单的配置和简洁的代码实现。

本文将对TomcatJdbc数据库连接池的初始化源码进行分析,Tomcat版本为9.0.82

正文

TomcatJdbc数据库连接池的初始化,发生在第一次获取数据库连接时,即第一次调用DataSourceProxy#getConnection方法的时候,如下所示。

public Connection getConnection() throws SQLException {
    // 一开始pool为null
    if (pool == null) {
        // 调用createPool()方法创建连接池
        return createPool().getConnection();
    }
    return pool.getConnection();
}

public ConnectionPool createPool() throws SQLException {
    if (pool != null) {
        return pool;
    } else {
        // 调用pCreatePool()方法创建连接池
        return pCreatePool();
    }
}

private synchronized ConnectionPool pCreatePool() throws SQLException {
    if (pool != null) {
        return pool;
    } else {
        // 调用数据库连接池的构造函数
        // 初始化逻辑在构造函数中
        pool = new ConnectionPool(poolProperties);
        return pool;
    }
}

public ConnectionPool(PoolConfiguration prop) throws SQLException {
    // 调用init()方法完成初始化
    init(prop);
}

在首次获取数据库连接时,由于ConnectionPoolnull,此时会new一个ConnectionPool作为数据库连接池并调用其init() 方法完成初始化,下面看一下init() 方法的实现。

protected void init(PoolConfiguration properties) throws SQLException {
    // 这里的properties实际是PoolProperties
    // 用户的数据库连接池配置会加载到properties中
    // 没有配置的项就使用PoolProperties中的默认值
    poolProperties = properties;

    // 做一下数据库连接池配置的常规校验
    // 例如maxIdle不能小于minIdle
    checkPoolConfiguration(properties);

    // busy队列保存借出的正在使用的连接
    busy = new LinkedBlockingQueue<>();
    if (properties.isFairQueue()) {
        // idle队列保存可用的空闲连接
        // 默认配置下会使用公平队列
        // 越先等待的线程越先获取到连接
        idle = new FairBlockingQueue<>();
    } else {
        idle = new LinkedBlockingQueue<>();
    }

    // 初始化连接池清理器,实际是启动一个定时任务
    // 每隔TimeBetweenEvictionRunsMillis运行一次
    initializePoolCleaner(properties);

    if (this.getPoolProperties().isJmxEnabled()) {
        createMBean();
    }

    // 解析得到并初始化拦截器
    ......

    // 根据配置的初始连接数initialSize来预热连接池
    // 若未配置initialSize则默认取值为10
    PooledConnection[] initialPool = new PooledConnection[poolProperties.getInitialSize()];
    try {
        for (int i = 0; i < initialPool.length; i++) {
            // 这里创建连接时,传入的用户名和密码为null不影响连接创建
            // 创建出来的连接会先暂时存放在initialPool数组里
            initialPool[i] = this.borrowConnection(0, null, null);
        }
    } catch (SQLException x) {
        log.error("Unable to create initial connections of pool.", x);
        if (!poolProperties.isIgnoreExceptionOnPreLoad()) {
            if (jmxPool!=null) {
                jmxPool.notify(org.apache.tomcat.jdbc.pool.jmx.ConnectionPool.NOTIFY_INIT, getStackTrace(x));
            }
            close(true);
            throw x;
        }
    } finally {
        for (int i = 0; i < initialPool.length; i++) {
            if (initialPool[i] != null) {
                try {
                    // 将预热出来的连接放到idle队列中
                    this.returnConnection(initialPool[i]);
                } catch(Exception x) {

                }
            }
        }
    }
    // 将连接池关闭标识置为false
    closed = false;
}

总结

通过init() 方法,我们可以总结如下几点。

  1. 连接池的默认配置在PoolProperties中。TomcatJdbc数据库连接池会读取用户的配置到PoolProperties中,如果有用户没有配置的项,则会使用PoolProperties预置的默认值,所以查看TomcatJdbc数据库连接池配置的默认值,可以在PoolProperties中查看;
  2. 连接池的连接存放在busyidle队列中。busy队列保存借出的正在使用的连接,idle队列保存可用的空闲连接,默认情况下,idle队列使用的是公平队列FairBlockingQueue,以确保最先等待获取连接的线程能最先获取到连接;
  3. 连接池在初始化时会创建连接池清理器PoolCleaner并启动。PoolCleaner本质是一个定时任务,每间隔timeBetweenEvictionRunsMillis运行一次,每次运行会执行连接泄漏检测可用连接检测保活idle队列大小调整
  4. 连接池会在初始化时预热。预热即预先创建指定数量的连接出来并放在idle队列中,预热连接数量通过initialSize指定,不指定时默认为10。


链接:https://juejin.cn/post/7306723921717428275

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

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

相关文章

OpenHarmony实战开发-基础知识

Svg组件主要作为svg画布的根节点使用&#xff0c;也可以在svg中嵌套使用。 说明&#xff1a; svg父组件或者svg组件需要定义宽高值&#xff0c;否则不进行绘制。 创建Svg组件 在pages/index目录下的hml文件中创建一个Svg组件。 <!-- xxx.hml --> <div class"co…

[华为OD] C卷 5G网络 现需要在某城市进行5G网络建设,已经选取N个地点设置5G基站 200

题目 现需要在某城市进行5G网络建设&#xff0c;已经选取N个地点设置5G基站&#xff0c;编号固定为1到N,接 下来需要各个基站之间使用光纤进行连接以确保基站能互联互通&#xff0c;不同基站之间架设光纤的成 本各不相同&#xff0c;且有些节点之间已经存在光纤相连&#…

力扣刷题 63.不同路径 II

题干 一个机器人位于一个 m x n 网格的左上角 &#xff08;起始点在下图中标记为 “Start” &#xff09;。 机器人每次只能向下或者向右移动一步。机器人试图达到网格的右下角&#xff08;在下图中标记为 “Finish”&#xff09;。 现在考虑网格中有障碍物。那么从左上角到…

(css)鼠标移出样式不变

(css)鼠标移出样式不变 需求&#xff1a;列表鼠标移入切换样式&#xff0c;移出保持不变 <divv-for"(item, index) of newsList":key"index"class"news-list":class"{active : change index}"tabindex"1"mouseenter&quo…

docker各目录含义

目录含义builder构建docker镜像的工具或过程buildkit用于构建和打包容器镜像&#xff0c;官方构建引擎&#xff0c;支持多阶段构建、缓存管理、并行化构建和多平台构建等功能containerd负责容器生命周期管理&#xff0c;能起、停、重启&#xff0c;确保容器运行。负责镜管理&am…

2024最新的,免费的 ChatGPT 网站AI(八个)

ChatGPT是美国人工智能研究实验室OpenAI在2022年11月推出的一款人工智能技术驱动的语言模型应用。它基于GPT-3.5架构&#xff08;后续还有GPT-4架构的升级版&#xff09;构建&#xff0c;拥有强大的自然语言处理能力和上下文理解能力&#xff0c;能够参与多轮对话&#xff0c;为…

恩智浦如何使用DITA

▲ 搜索“大龙谈智能内容”关注公众号▲ 作者 | John Walker - NXP销售和市场营销业务分析师 2013年4月18日 作为恩智浦半导体公司销售和市场部的业务分析师&#xff0c;我负责恩智浦半导公司产品信息的数据/内容模型、流程和工具。我来自英国&#xff0c;但自2000年以来一…

Python3 循环语句

Python 中的循环语句有 for 和 while。 Python 循环语句的控制结构图如下所示&#xff1a; while 循环 Python 中 while 语句的一般形式&#xff1a; while 判断条件(condition)&#xff1a;执行语句(statements)…… 执行流程图如下&#xff1a; 同样需要注意冒号和缩进。…

go语言实现简单登陆返回token样例

目录 1、代码实现样例&#xff1a; 2、postman调用&#xff0c;获取登陆后的token&#xff1a; 1、代码实现样例&#xff1a; package mainimport ("net/http""time""github.com/dgrijalva/jwt-go""github.com/gin-gonic/gin" )var …

Leetcode—2639. 查询网格图中每一列的宽度【简单】

2024每日刷题&#xff08;121&#xff09; Leetcode—2639. 查询网格图中每一列的宽度 实现代码 class Solution { public:int func(int num) {if(num 0) {return 1;}int len 0;while(num ! 0) {len;num / 10;}return len;}vector<int> findColumnWidth(vector<ve…

怎么通过isinstance(Obj,Class)验证?【isinstance】

最近有这样一个项目&#xff0c;这个项目可以用一个成熟的项目的构造树&#xff0c;读取树&#xff0c;再检索的过程&#xff0c;现在有新的需求&#xff0c;另一个逻辑构造同样节点结构的树&#xff0c;pickle序列化保存&#xff0c;再使用原来项目的读取、检索函数&#xff0…

Altair® PBS Professional®——行业超前的 HPC 和高吞吐量计算工作负载管理器和作业调度程序

PBS Professional 是一款快速、强大的工作负载管理器&#xff0c;旨在提高生产力、优化利用率和效率&#xff0c;并简化集群、云和超级计算机的管理——从极大的 HPC 工作负载到数百万个小型、高吞吐量作业。PBS Professional 能够自动执行作业调度、管理、监视和报告任务&…

4月25日 C++day3

#include <iostream> using namespace std;class Person {const string name;int age;char sex; public:Person():name("lisi"){cout << "Person无参构造" << endl;}Person(string name,int age,char sex):name(name),age(age),sex(sex)…

WordPress内存不足如何处理

本周有一个客户&#xff0c;购买Hostease的Linux虚拟主机&#xff0c;询问我们的在线客服&#xff0c;站点出现WordPress内存不足如何处理。我们为用户提供教程&#xff0c;用户很快完成了设置。在此&#xff0c;我们分享这个操作教程&#xff0c;希望可以对您有帮助。 WordPre…

Sora新突破!AI生成电影迈向新阶段,配音版Sora登场!将如何改变影视行业?

Sora之后迎来新突破&#xff01; 配音版Sora来袭&#xff0c;AI生成电影又更近一步&#xff01; 在2024年伊始&#xff0c;人工智能界迎来了一次创新性的突破&#xff0c;由AI语音技术的先锋公司ElevenLabs带头实现。他们最近的成就体现在为OpenAI的Sora视频模型提供了令人动容…

k8s学习(三十七)centos下离线部署kubernetes1.30(高可用)

文章目录 准备工作1、升级操作系统内核1.1、查看操作系统和内核版本1.2、下载内核离线升级包1.3、升级内核1.4、确认内核版本 2、修改主机名/hosts文件2.1、修改主机名2.2、修改hosts文件 3、关闭防火墙4、关闭SELINUX配置5、时间同步5.1、下载NTP5.2、卸载5.3、安装5.4、配置5…

Leetcode—1041. 困于环中的机器人【中等】

2024每日刷题&#xff08;121&#xff09; Leetcode—1041. 困于环中的机器人 实现代码 class Solution { public:bool isRobotBounded(string instructions) {int x 0;int y 0;int d 0;vector<vector<int>> direction{{0, 1}, {1, 0}, {0, -1}, {-1, 0}};for…

[嵌入式系统-54]:RT-Thread:内核基础与核心概念

目录 前言&#xff1a; 一、RT-Thread 内核介绍 1.线程调度 2.时钟管理 3.线程间同步与互斥 4.线程间通信 5.内存管理 6.I/O 设备管理 二、RT-Thread 启动流程 三、RT-Thread 程序内存分布 四、RT-Thread 自动初始化机制 五、RT-Thread 内核对象模型 1. 静态对象和…

ElasticSearch总结1

目录 一、ElasticSearch介绍&#xff1a; 举例一&#xff1a; 举例二&#xff1a; 举例三&#xff1a; 二、ELK技术栈 三、Elasticsearch 的基本概念&#xff1a; 四、正向索引和倒排索引&#xff1a; 正向索引&#xff1a; 倒排索引&#xff1a; 五、Mysql和Elastics…

http1.1和http2.0的同源请求数限制

判断协议版本 :scheme: 在请求头中表示使用的是HTTP/2协议。即 出现 :开头的请求头Chrome 只支持查看 HTTP/1.x 的 Raw Headers&#xff0c;对这种请求&#xff0c;会给出 view source 选项。HTTP2.0不给出。可继续学习 https://www.cnblogs.com/kirito-c/p/10360868.html抓包…
最新文章