专栏首页Java技术栈数据库连接池为啥要用 ThreadLocal?

数据库连接池为啥要用 ThreadLocal?

这个问题我疑问了很久很久,主要如下截图。

一个连接对应一个事务,多个连接的事务是不一样的,先大概了解一下,往下看?

本人是在学threadlocal的时候,网上大部分人都是说数据库连接池是典型的用了threadloca的例子,然后我就又查数据库连接池和threadloca的关系。

但是,99%都说threadlocal是为了在并发的情况下,为了保证线程安全,创建了副本什么的,其实这只是threadlocal的用法之一,它还有个用法就是确保同一线程之间参数传递的方便(扯远了)

回归正题,还是拿上面的图来说事。。。。

我只讲两个关键点,明白人一看就懂:

1、两者有根本性的区别,用处不一样!

连接池是缓存并托管数据库连接,主要是为了提高性能。

而ThreadLocal缓存连接,是为了把同一个数据库连接“分享”给同一个线程的不同调用方法。(不管调用哪个方法,都是使用的同一个连接,方便进行“跨方法”的事务控制)

举个栗子:

如果一个请求中涉及多个 DAO 操作,而如果这些DAO中的Connection都是独立的话,就没有办法完成一个事务。但是如果DAO 中的 Connection 是从 ThreadLocal 中获得的(意味着都是同一个对象), 那么这些 DAO 就会被纳入到同一个 Connection 之下。

2、重点要理解“连接池”。

连接池里面有一定数量的连接资源,比如最大20个连接。Spring Boot 学习笔记,推荐学习下。

题外话:如果直接通过 Java原生API 获取“直连”的话:

底层方法一般都是这样写的:

java.sql.DriverManager.getConnection(url, props);

java.sql.Driver.connect(url, props);

特点是:要传入url、用户名和密码等信息)

这种方式,肯定就没有使用数据库连接池。

使用数据库连接池,通常都是得到一个所谓的javax.sql.DataSource[接口]的实例对象,它里面包含了Connection,并且数据库连接池工具类(比如C3P0、JNDI、DBCP等),肯定是重新定义了getConnection、closeConnection等方法。

所以你每次得到的Connection,几乎都不是新建立的连接(而是已经建立好并放到缓存里面的连接),你调用closeConnection方法,也不是真正的关闭连接(一般都是起到一个标识作用,标识当前连接已经使用完毕,归还给连接池,让这个连接处于待分配状态)

PS:所以说:使用数据库连接池时,还是要显式的调用数据库连接池API提供的关闭连接的方法。

理解一下这句话:2021 最新 Java 面试题出炉!

不同的线程在同一个时间( 或者 同一个线程在多个地方)从连接池中拿到的Connection,肯定不是同一个连接。(反过来讲:不同时间的两个线程,一前一后,则有可能拿到同一个连接)

总结:

再好好理解一下上面的一段话,我再最后解释亿下。。。

首先,我们为了避免单一数据库连接的创建和关闭耗费时间和性能,引入了数据库连接池,提前创建好了n条连接放入池中,如果是单线程情况下,那这样挺好的。

那如果是多线程情况下呢?

还是上面那段话,假设同一时间多个线程从数据库连接池获取连接,那肯定拿的是不同的连接,我当前线程和别的线程拿的连接不一样,那我当前在crud的时候,不在一个事务之内。

假设不同时间的多个线程要从数据库连接池拿连接,那这个时候就可能拿到的是同一个连接了,那我多个线程线程拿到的是同一个连接,也就是说在多个线程在同一个事务之内,线程a执行了插入还没来得及提交,线程b此时来了个更新,在线程a还未操作完之前,线程b更新完了后,直接把连接给close了,线程a插了一半发现插不了了。。。

为了确保不同时间多个线程可能拿到的是同一个连接,那么此时threadlocal闪亮登场,就算我拿的是“同一个连接”,在引入了threadlocal后,每个线程之间都会创建独立的连接副本,将collection各自copy一份,这样就互相不干扰了。

以上是我的个人见解。另外,关注公众号Java技术栈,在后台回复:面试,可以获取我整理的 Java 多线程系列面试题和答案,非常齐全。

原文链接:https://blog.csdn.net/qq_42405666/article/details/108258820

版权声明:本文为CSDN博主「奔跑的乌龟」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。

本文分享自微信公众号 - Java技术栈(javastack),作者:点击关注 ????

原文出处及转载信息见文内详细说明,如有侵权,请联系 yunjia_community@tencent.com 删除。

原始发表时间:2021-06-10

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

我来说两句

0 条评论
登录 后参与评论

相关文章

  • 数据库连接池为什么要用threadlocal呢?(不用会怎样?)

    我先说为什么引入threadlocal,其实是为了解决数据库事务,而事务是和连接有关的,每个连接对应一个事务,多个连接的事务是不一样的,先大概了解一下,往下看?

    架构师修炼
  • 使用连接池连接数据库

    我在网上看到的一个教程,感觉那个老师总结的特别好,他是引用<红楼梦>中的人物,将连接数据库的步骤进行了总结。 “贾琏欲执事” 贾:加载注册驱动。 琏:连接...

    微醺
  • PostgreSQL 为什么接受大量连接到数据库需要连接池

    PostgreSQL 是非常好的开源的数据库,主要针对替换ORACLE及其他传统型RDBS数据库的重任,基本上大部分中小型企业,能指望的开源数据库也只有POST...

    AustinDatabases
  • 数据库连接池

    数据库连接池相关资料: 关于数据库连接池的使用,首先我们要明白我们为什么要用它,对应普通的数据库连接操作,通常会涉及到以下一些操作是比较耗时的: 网络通讯...

    Steve Wang
  • 数据库连接池

    HTTP Status 500 - Could not open connection type Exception report message Could ...

    yawn
  • 数据库连接池

    官方:数据库连接池(Connection pooling)是程序启动时建立足够的数据库连接,并将这些连接组成一个连接池,由程序动态地对池中的连接进行申请,使用,...

    Java架构师历程
  • 数据库连接池

    数据库连接对象是有限资源,所以数据库连接池是用于负责分配、管理和释放数据库连接对象,它允许应用程序重复使用一个现有的数据库连接对象,而不是再重新建立一个;这一点...

    端碗吹水
  • 基于Swoole的通用连接池 - 数据库连接池

    open-smf/connection-pool 是一个基于Swoole的通用连接池,常被用作数据库连接池。

    猿哥
  • 数据库连接池简单连接

    模型      -----     Model 视图      -----     View 控制器  -----     Controller

    qubianzhong
  • 数据库连接池DBUtils使用

      DBUtils简单说python实现的线程化数据库连接(连接池),DBUtils支持所有遵循DP-API 2规范的数据库连接模块,例如:mysql、sqls...

    py3study
  • 常用Java数据库连接池

    在这里所谓的数据库连接是指通过网络协议与数据库服务之间建立的TCP连接。通常,与数据库服务进行通信的网络协议无需由应用程序本身实现,原因有三:

    编程随笔
  • JNDI数据库连接池

    JNDI的全称是java命名与目录接口(Java Naming and Directory Interface),是一个应用程序设计的API,为开发人员提供了查...

    汤高
  • DBUtils数据库连接池

    使用数据库连接池技术,可以重复使用多个数据库连接,避免每次执行数据库操作都建立连接和关闭连接,也避免了大型应用同时占用多个数据库连接。

    py3study
  • 头条面试官手把手教学 ThreadLocal

    我们看下JDK文档的官方描述:ThreadLocal类用来提供线程内部等局部变量,这种变量在多线程环境下访问(get,set)时能保证各个线程的变量相对独立于其...

    sowhat1412
  • 数据库连接(2) - 为什么C3P0连接池那么慢

    在上一篇中我们介绍说客户端建立一次连接耗时太长(建立连接,设置字符集,autocommit等),如果在每个sql操作都需要经历建立连接,关闭连接。不仅应用程序响...

    方丈的寺院
  • C3p0数据库连接池的使用

    1:首先介绍c3p0的一般使用用法,这种用法练习使用,当然工作的时候使用第二种xml配置完成c3p0的使用。 c3p0的网址:http://www.mchang...

    别先生
  • 数据库连接池为什么首选Druid

    Druid是阿里巴巴的一个开源项目,号称为监控而生的数据库连接池,在功能、性能、扩展性方面都超过其他,例如 DBCP、C3P0、BoneCP、Proxool、J...

    用户7676729
  • JDBC(九)数据库连接池

    ①、普通的JDBC数据库连接使用 DriverManager 来获取,每次向数据库建立连接的时候都要将 Connection 加载到内存中,再验证用户名和密码(...

    leeqico
  • Python数据库连接池DBUtils

     如果没有连接池,使用pymysql来连接数据库时,单线程应用完全没有问题,但如果涉及到多线程应用那么就需要加锁,一旦加锁那么连接势必就会排队等待,当请求比较多...

    新人小试

扫码关注云+社区

领取腾讯云代金券