软件设计思想:池化技术

简介: 软件设计思想:池化技术

软件设计思想:池化技术



先举个例子


一种简单的架构:前端一台 Web 服务器运行业务代码,后端一台数据库服务器存储业务数据。


640.png


系统上线时,用户量不大,运行平稳,但是随着流量的提升,系统访问速度会变慢。

分析日志后,发现系统慢的原因是出现在系统库的交互上,因为数据调用的方式是首先获取数据库的连接,然后依靠这个连接,查询数据库的数据,最后释放数据库资源。


为什么频繁创建连接响应会变慢?


整个 MySQL 创建连接过程主要分成两个部分:

  1. 第一部分是采用建立 TCP 连接。通过三次握手 建立 TCP 连接。
  • 客户端向服务端发送一个 SYN 包
  • 服务端给客户端发送一个 ACK  包和一个 SYN 包
  • 第三个是客户端给服务端发送一个 ACK  包
  1. 第二部分是 MySQL 服务端校验客户端密码的过程

第一个包是服务端给客户端要认证的报文,第二个和第三个包是客户端加密后发送给服务端的包,最后两个包是服务端给客户端 OK  的报文。

数据校验过程大概占整个查询过程的80% 时间,比如 数据校验占 4ms ,查询过程使用 1ms 这样频繁创建连接,创建连接严重影响了性能。


如何优化呢?


如果数据库不再频繁创建连接,如果实现就使用连接池,将连接先建立好,这样不频繁创建连接,那么查询性能会大大提升。

用连接池预先建立数据库连接

开发过程中往往会用到很多连接池,比如数据库连接池, HTTP 连接池,Redis 连接池等。


数据库连接池配置


  • 如果当前连接数小于最小连接数,则创建新的数据库请求;
  • 如果连接池中有空闲连接则复用空闲连接
  • 如果空闲池中没有连接并且连接数小于最大连接数,则创建新的连接处理请求;
  • 如果当前连接数已经大于等于最大连接数,则按照配种中设定的时间(c3p0 checkoutTimeOut)等待旧连接可用。
  • 如果等待超过这个设定值则向用户抛出异常。

看个实例:


<bean id="dataSource" class="org.apache.commons.dbcp.BasicDataSource"
          destroy-method="close">
        <property name="driverClassName" value="oracle.jdbc.driver.OracleDriver"/>
        <property name="url" value="xxxx"/>
        <property name="username" value="xxxx"/>
        <property name="password" value="xxxx"/>
        <!-- 连接池最大数量 --> 
        <property name="maxActive" value="60"/>
        <!-- 初始化连接大小 -->
        <property name="initialSize" value="5"/>
         <!-- 连接池最小空闲 -->  
        <property name="minIdle" value="5"/>
        <!-- 连接池最大空闲 -->  
        <property name="maxIdle" value="10"/>
        <!-- 获取连接最大等待时间 -->  
        <property name="maxWait" value="60000"/>
        <property name="testOnBorrow" value="true"/>
        <!-- 是否对sql进行自动提交-->
        <property name="defaultAutoCommit" value="true"/>
        <!-- 读写1分钟超时 -->
        <property name="connectionProperties" value="oracle.jdbc.ReadTimeout=60000"/> 
        <property name="validationQuery" value="SELECT 1 FROM dual"/>
    </bean>


"select 1"  的命令用来检查连接是否可用。

testOnBorrow 配置项 是先校验连接是否可用,如果可用采用执行 SQL 语句。这种连接方式会引入多余的开销,线上可用尽量不要使用,在测试服务上可用


  • initialSize,连接初始值,连接池启动时创建的连接数量的初始值
  • maxActive,连接池的最大值,同一时间可以从池分配的最多连接数量,0时无限制
  • maxIdle,最大空闲值.当经过一个高峰时间后,连接池可以慢慢将已经用不到的连接慢慢释放一部分,一直减少到maxIdle为止 ,0时无限制
  • minIdle,最小空闲值.当空闲的连接数少于阀值时,连接池就会预申请去一些连接,以免洪峰来时来不及申请
  • poolPreparedStatements,是否对已备语句进行池管理(布尔值),是否对PreparedStatement进行缓存
  • defaultAutoCommit,是否对sql进行自动提交,进行事务管理的时候往往要关闭jdbc的自动提交功能。

假设有一个接口,需要三次访问数据库,但是根据经验判断,觉得这个可能成为系统瓶颈,为避免频繁创建线程导致的开销,可以使用线程池来管理多个线程与数据库的交互


用线程池创建线程


JDK 1.5 中引入的 ThreadPoolExecutor 就是一种线程池的实现,有几个重要的参数:coreThreadPool 和 maxThreadCount,这两个参数控制整个线程池的执行过程。

  • 当线程池中的线程数小于 coreThreadCount 时,处理新的任务是会创建新的线程。
  • 如果线程数大于 coreThreadCount ,则把任务丢到一个队列中,当有空闲线程再执行。
  • 当队列中的任务队列满了的时候,则继续创建线程,直到达到 maxThreadCount。
  • 当线程数达到 maxThreadCount 时,还有新的任务提交,那么我们不得不将他们丢弃。


640.png


实际项目中曾经有这么个问题,任务丢给线程池后,长时间都没有被执行,后来排查发现是 coreThreadCount 和 maxThreadCount 设置的比较小。


池化技术


池化技术的思想: 核心思想是空间换时间,期望使用预先创建好的对象来减少频繁创建对象的性能开销,同时还可以对对象进行统一管理,减少对象使用成本。


总结


  • 池子的最大值和最小值设置需要根据实际运行情况做调整
  • 池子中的对象在需要在使用前预先初始化,比如线程池先创建核心线程。
  • 池化技术的核心思想是空间换时间,避免出现空间过度使用出现内存泄露或者频繁垃圾回收等问题。
相关文章
|
6月前
|
架构师 测试技术 Linux
嵌入式软件架构中抽象层设计方法
嵌入式软件架构中抽象层设计方法
226 0
|
5月前
|
Java 持续交付 数据库
避免服务分层污水池反模式
【6月更文挑战第30天】本文介绍污水池反模式,分层架构在敏捷性、部署性和性能方面得分较低,但具有高测试性和易开发性。关键在于合理分层以降低耦合和提高解耦效果。
257 1
避免服务分层污水池反模式
|
3月前
|
程序员
软件设计与架构复杂度问题之战略编程与战术编程的主要区别如何解决
软件设计与架构复杂度问题之战略编程与战术编程的主要区别如何解决
|
5月前
|
架构师 持续交付 微服务
探索软件架构设计的深层逻辑
【6月更文挑战第5天】在数字化浪潮中,软件架构设计如同搭建一座虚拟的巴别塔,它不仅需要承载技术的重量,还要预见未来的需求。本文将通过我的个人经验,探讨如何在变化莫测的技术海洋中,寻找到稳固的架构基石,以及如何让这座塔楼灵活地适应不断变化的环境。
46 1
|
6月前
|
存储 安全 关系型数据库
技术人必修课:利用金字塔原理高效思考与表达
作者写这篇文章的目的就是希望能够帮助更多同学了解金字塔原理并合理应用,不只是写作,更是要着眼于思考和表达。本文将围绕认识金字塔结构、表达的逻辑、思考的逻辑、解决问题的框架、演示的逻辑这几个方面带领大家深入学习金字塔原理。
|
6月前
|
设计模式 算法 Java
深入理解面向对象设计的深层原则与思维
软件设计原则是指在软件开发过程中,通过一系列指导性的原则来指导设计决策和编码实践。这些原则旨在提高软件系统的质量,使其具有可维护性、可扩展性、可重用性和可测试性。几个重要性:可维护性、可扩展性、可重用性、可测试性和降低系统复杂度。软件设计原则是提高软件系统质量和可维护性的基石。遵循这些原则可以使得代码更加清晰、灵活和可靠,提高开发效率和软件质量,减少后期维护成本。同时,它们也为团队合作和团队成员共同理解代码提供了共同的规范和指导。
143 2
深入理解面向对象设计的深层原则与思维
|
6月前
|
存储 自然语言处理 前端开发
软考实践之分层架构思想的理论和应用实践
软考实践之分层架构思想的理论和应用实践
301 0
【逻辑思维训练 一】金字塔思维概述
【逻辑思维训练 一】金字塔思维概述
126 0
|
存储 缓存 架构师
计算机架构设计的 8 个伟大思想
计算机架构设计的 8 个伟大思想
|
SQL Cloud Native 数据可视化
模块化思想在实践中的应用
各种编程语言中的函数,数据仓库的标签体系,甚至于数据中台的核心理念,都是把模块化的思想发挥到了极致,避免了我们重复造轮子,消除了数据烟囱,用最小的投入获得了最大的产出。
模块化思想在实践中的应用