灵魂一问:数据库连接池到底该怎么配?

来自公#众#号:新世界杂货铺

好家伙,我直接好家伙!GitHub不愧是全球最大的同性交友网站,资源丰富且质量高!

连接池的配置应该按照什么原则来?这个问题在笔者心中疑惑良久,直到在GitHub上发现了About Pool Sizing这篇文章。看完之后一扫笔者心中阴霾,神清气爽。妈妈再也不用担心我在项目中瞎配连接池啦!

以下内容为笔者根据原文翻译总结所得。

原文提到开发人员经常会将连接池配置错误,而要想正确配置连接池需要理解一些原则,即使这些原则可能违反人类直觉。

1万并发用户访问

假设你有一个需要每秒处理2万事务的网站,你的连接池应该配置多大?令人惊讶的是,这个问题不应该是连接池配置多大而是应该配置多小。

下面这个视频是Oracle Real-World Performance组发布的(笔者建议各位读者架个梯子亲自看看):

https://www.youtube.com/watch?v=xNDnVOCdvQ0

下面笔者对视频中的内容进行简单的概括。

注:视频中是对Oracle数据库进行测试,笔者在开发/线上环境均使用Mysql数据库,但触类旁通,该测试结果仍具有很高的参考价值。

压测初始配置如下:

并发线程数 9600
每两次访问数据库之间Sleep 550ms
初始线程池大小 2048

笔者随机截取视频中第一次压测结果如下:

image

Sessions达到2048时,Queue-ms(每个请求在队列中的等待时间)为30ms,Run-ms(SQL执行时间)为71ms,同时还有很多buffer busy waits

接下来其他条件不动,仅将初始线程池大小设置为1024得到如下结果:

image

Sessions达到1024时,Queue-ms(38ms)和第一次相比可以认为几乎没有变化,Run-ms(30ms)和第一次相比明显减少,buffer busy waits和第一次相比也明显减少。

最后,将线程池大小设置为96,其他条件不变得到如下结果:

image

此时,Queue-ms和Run-ms耗时极短并且看不到任何的buffer busy waits。好家伙,原作者对此时的结果直接打了一个✅。

最后,一起对比一下三次压测结果的吞吐量:

image

图中上半部分为耗时因子,下半部分为吞吐量。红框、蓝框和黄框分别代表着链接池大小为2048、1024和96的吞吐量。由折线图知当连接池大小为96时吞吐量明显上升。

没有做任何其他调整,仅减小连接池大小就可将应用程序的性能提升近50倍!

But why?

为什么nginx只用4个线程发挥出的性能就大大超越了100个进程的Apache Web服务器?回想一下计算机科学的基础知识,答案其实是很明显的。

即使是单核的计算机也可以“同时”支持数十个或数百个线程。但是我们都应该知道,这仅仅是操作系统通过时间分片交替执行的一个小把戏,实际上,单个内核一次只能执行一个线程,然后操作系统切换上下文执行另一个线程的代码,依此类推。给定一颗CPU核心,其顺序执行AB永远比通过时间分片“同时”执行A和B要快,这是一条计算机科学的基本法则。一旦线程的数量超过了CPU核心的数量,再增加线程数系统就只会更慢,而不是更快。

笔者认为上述A,B在没有I/O阻塞时,顺序执行才比同时执行更快。

有限的资源

当我们排查数据库的性能瓶颈时,它们可以概括为三个类别:CPU,磁盘,网络。内存和磁盘、网络相比,带宽高出好几个数量级故忽略此排查方向。

如果我们忽略磁盘和网络,那就更加简单了。在一个8核的服务器上,将连接数设置为8将提供最佳性能,再增加连接数就会因上下文切换的损耗导致性能下降。但是我们不能忽视磁盘和网络。

数据库通常将数据存储在磁盘上,对于老式的机械硬盘存在寻址时间成本和旋转时间成本。在这段时间内( I/O等待),连接/查询/线程被阻塞以等待磁盘,此时操作系统控制CPU执行其他线程代码以更好地利用CPU资源。所以,由于线程在I/O上阻塞,我们可以让线程/连接数比CPU核心多一些,这样能够在同样的时间内完成更多的工作。

那连接数具体应该设置多少呢?这取决于磁盘。因为新型的SSD不需要寻址和旋转开销。此时不要想当然地认为“SSD速度更快,所以我们应该有更多的线程数”,恰好相反,更快意味着更少的阻塞,因此越接近核心数量的线程将会发挥最优的性能。只有当阻塞创造了更多的执行机会时,更多的线程数才能发挥出更好的性能

网络类似于磁盘。当发送/接收缓冲区填满并停止时,通过以太网接口在线路写入数据也会导致阻塞。 10G宽带延迟小于1G宽带,而1G宽带的延迟又小于100M宽带。就阻塞而言,网络通常是放在第三位考虑的,但是仍然有人会在性能计算中忽略它。

说实话,下面这张图笔者研究了半天也不明白它的意义,不过既然原文贴出来了,笔者只好照搬不误。

image

在上述PostgreSQL基准测试中可以看到,TPS速率在大约50个连接处开始趋于平稳。 在上面的Oracle视频中,将连接数从2048下调至96。但实际上,96也很高了,除非服务器使用的是16或32核的处理器。

计算公式

虽然下面的公式是PostgreSQL提供的,不过我们认为该公式可以广泛地适用于不同的数据库。你可以利用该公式计算一个初始值,以此初始值为基准模拟负载并调整连接数大小从而找到一个合适的连接数。

连接数 = ((核心数 * 2) + 有效磁盘数)

在多年的基准测试中,保持最优吞吐量的活跃连接数都是接近该公式的计算结果。 核心数不应包含超线程,即使启用了超线程也是如此。如果活跃数据全部被缓存了,那么有效磁盘数是0,随着缓存命中率的下降,有效磁盘数将逐渐趋近于实际的磁盘数。

特别注意:目前为止,还没有任何关于该公式作用于SSD的效果分析。

这个公式意味着什么?假如你有一个服务器,该服务器具有一块磁盘和一个4核的i7处理器,那么此服务器的连接池大小应该是:9 =((4 * 2)+1),取个整数的话就是10。是不是看起来很小?但是请尝试一下,我们敢打赌在这样的设置下,它可以轻松搞定3000个前端用户以6000TPS的速率执行简单查询。如果你增加连接池的大小并运行负载测试,你会发现前端响应时间增加的同时TPS速率开始下降。

公理:你需要一个小的充满了等待连接的线程队列

如果你有10000个用户,设置一个10000的连接池基本等于疯了,1000仍然很恐怖,即使是100也太多了。你最多需要一个十几个连接的小型池,其余的业务线程则被阻塞直到有可用连接。连接池中的连接数量应该等于你的数据库能够有效同时进行的查询任务数(通常不会高于2*CPU核心数)。

我们经常见到一些小规模的web应用,应付着大约十来个的并发用户,却使用着一个100连接数的连接池。不要过度配置数据库。

笔者想到一个案例:曾经有一个读者将HTTP连接池的最大空闲连接设置为300,后来发现经常出问题,最后在笔者的劝说下使用了默认配置之后就再也没找过我。

避免死锁的连接池大小

单个线程同时需要多个连接可能会造成死锁。这在很大程度上是一个业务上的问题,该问题可以通过增加连接池大小来解决。但是在增加连接池大小之前,我们还是强烈建议您首先检查在业务方面可以做什么。

为避免死锁,计算连接池有一个简单的计算公式:

连接数 = 最大线程数 * (单线程需要的最大连接数 - 1) + 1

假如你有3个线程,每个线程需要4个连接来执行某些任务。确保永不发生死锁所需的池大小为:

3 x (4-1) + 1 = 10

👉这不一定是最佳的连接池大小,而是避免死锁所需的最小的连接池大小。

笔者根据自己的开发经验在此特意提醒:

  1. Go中单个协程尽量不要同时使用多个连接进行操作。
  2. 执行事务期间不要通过非当前事务连接获取数据, 请使用当前事务连接直接获取数据(如果在执行事务期间使用非当前事务连接获取数据相当于同时使用两个连接)。

忠告

连接池的大小最终与系统特性有关。

例如,一个混合了长时事务和短事务的系统是非常难以使用连接池进行调优的。通常做法是, 使用两个连接池,一个用于长时事务,一个用于实时查询。

在主要运行长时事务的系统中,连接池大小通常存在外部约束。例如,一个任务执行队列只允许固定数量的任务同时运行。此时,任务队列的大小应该去适应连接池的大小,而不是反过来。

最后,衷心希望笔者的翻译能够对各位读者有一定的帮助。

©著作权归作者所有,转载或内容合作请联系作者
禁止转载,如需转载请通过简信或评论联系作者。
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,384评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,845评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,148评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,640评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,731评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,712评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,703评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,473评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,915评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,227评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,384评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,063评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,706评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,302评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,531评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,321评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,248评论 2 352

推荐阅读更多精彩内容