DBUtils 是Python 的一个用于实现数据库连接池的模块。

此连接池有两种连接模式:

DBUtils :提供两种外部接口:

PersistentDB :提供线程专用的数据库连接,并自动管理连接。

PooledDB :提供线程间可共享的数据库连接,并自动管理连接。

介绍

PersistentDB模式

为每个线程创建一个连接,线程即使调用了 close 方法,也不会关闭,只是把链接重新放到链接池,供自己线程再次使用,当线程终止时,链接自动关闭。

  1. from DBUtils.PersistentDB import PersistentDB
  2. import pymysql
  3. POOL = PersistentDB(
  4. creator=pymysql, # 使用链接数据库的模块
  5. maxusage=None, # 一个链接最多被重复使用的次数,None表示无限制
  6. setsession=[], # 开始会话前执行的命令列表。
  7. ping=0, # ping MySQL服务端,检查是否服务可用。
  8. closeable=False, # 如果为False时, conn.close() 实际上被忽略,供下次使用,再线程关闭时,才会自动关闭链接。如果为True时, conn.close()则关闭链接,那么再次调用pool.connection时就会报错,因为已经真的关闭了连接(pool.steady_connection()可以获取一个新的链接)
  9. threadlocal=None, # 本线程独享值得对象,用于保存链接对象,如果链接对象被重置
  10. host='127.0.0.1',
  11. port=3306,
  12. user='root',
  13. password='',
  14. database='test',
  15. charset='utf8'
  16. )
  17.  
  18. def func():
  19. conn = POOL.connection(shareable=False)
  20. cursor = conn.cursor()
  21. cursor.execute('select * from user')
  22. result = cursor.fetchall()
  23. print(result)
  24. cursor.close()
  25. conn.close()
  26. if __name__ == '__main__':
  27.  
  28. func()

PooledDB模式

创建一批连接到连接池,供所有线程共享使用。

  1. import pymysql
  2.  
  3. from DBUtils.PooledDB import PooledDB
  4. POOL = PooledDB(
  5. creator=pymysql, # 使用链接数据库的模块
  6. maxconnections=6, # 连接池允许的最大连接数,0和None表示不限制连接数
  7. mincached=2, # 初始化时,链接池中至少创建的空闲的链接,0表示不创建
  8. maxcached=5, # 链接池中最多闲置的链接,0和None不限制
  9. maxshared=3, # 链接池中最多共享的链接数量,0和None表示全部共享。PS: 无用,因为pymysql和MySQLdb等模块的 threadsafety都为1,所有值无论设置为多少,_maxcached永远为0,所以永远是所有链接都共享。
  10. blocking=True, # 连接池中如果没有可用连接后,是否阻塞等待。True,等待;False,不等待然后报错
  11. maxusage=None, # 一个链接最多被重复使用的次数,None表示无限制
  12. setsession=[], # 开始会话前执行的命令列表。
  13. ping=0, # ping MySQL服务端,检查是否服务可用。
  14. host='127.0.0.1',
  15. port=3306,
  16. user='root',
  17. password='',
  18. database='test',
  19. charset='utf8'
  20. )
  21.  
  22. def func():
  23. # 检测当前正在运行连接数的是否小于最大链接数,如果不小于则等待或报raise TooManyConnections异常
  24. # 否则则优先去初始化时创建的链接中获取链接 SteadyDBConnection。
  25. # 然后将SteadyDBConnection对象封装到PooledDedicatedDBConnection中并返回。
  26. # 如果最开始创建的链接没有链接,则去创建一个SteadyDBConnection对象,再封装到PooledDedicatedDBConnection中并返回。
  27. # 一旦关闭链接后,连接就返回到连接池让后续线程继续使用。
  28. conn = POOL.connection()
  29. cursor = conn.cursor(pymysql.cursors.DictCursor)
  30. cursor.execute('select * from user')
  31. result = cursor.fetchall()
  32. print(result)
  33. conn.close()
  34.  
  35. if __name__ == '__main__':
  36.  
  37. func()

工具类

  1. # TEST数据库信息
  2. DB_TEST_HOST = "127.0.0.1"
  3. DB_TEST_PORT = 3306
  4. DB_TEST_DBNAME = ""
  5. DB_TEST_USER = "root"
  6. DB_TEST_PASSWORD = "root"
  7.  
  8. # 数据库连接编码
  9. DB_CHARSET = "utf8"
  10.  
  11. # mincached : 启动时开启的闲置连接数量(缺省值 0 开始时不创建连接)
  12. DB_MIN_CACHED = 10
  13.  
  14. # maxcached : 连接池中允许的闲置的最多连接数量(缺省值 0 代表不闲置连接池大小)
  15. DB_MAX_CACHED = 10
  16.  
  17. # maxshared : 共享连接数允许的最大数量(缺省值 0 代表所有连接都是专用的)如果达到了最大数量,被请求为共享的连接将会被共享使用
  18. DB_MAX_SHARED = 20
  19.  
  20. # maxconnecyions : 创建连接池的最大数量(缺省值 0 代表不限制)
  21. DB_MAX_CONNECYIONS = 100
  22.  
  23. # blocking : 设置在连接池达到最大数量时的行为(缺省值 0 或 False 代表返回一个错误<toMany......>; 其他代表阻塞直到连接数减少,连接被分配)
  24. DB_BLOCKING = True
  25.  
  26. # maxusage : 单个连接的最大允许复用次数(缺省值 0 或 False 代表不限制的复用).当达到最大数时,连接会自动重新连接(关闭和重新打开)
  27. DB_MAX_USAGE = 0
  28.  
  29. # setsession : 一个可选的SQL命令列表用于准备每个会话,如["set datestyle to german", ...]
  30. DB_SET_SESSION = None

db_config.py

  1. import pymysql
  2. from DBUtils.PooledDB import PooledDB
  3. import db_config as Config
  4.  
  5. class ConnectionPool(object):
  6. __pool = None
  7.  
  8. def __enter__(self):
  9. self.conn = self.__getConn()
  10. self.cursor = self.conn.cursor()
  11. print("数据库创建conn和cursor")
  12. return self
  13.  
  14. def __getConn(self):
  15. if self.__pool is None:
  16. self.__pool = PooledDB(creator=pymysql, mincached=Config.DB_MIN_CACHED, maxcached=Config.DB_MAX_CACHED,
  17. maxshared=Config.DB_MAX_SHARED, maxconnections=Config.DB_MAX_CONNECYIONS,
  18. blocking=Config.DB_BLOCKING, maxusage=Config.DB_MAX_USAGE,
  19. setsession=Config.DB_SET_SESSION,
  20. host=Config.DB_TEST_HOST, port=Config.DB_TEST_PORT,
  21. user=Config.DB_TEST_USER, passwd=Config.DB_TEST_PASSWORD,
  22. db=Config.DB_TEST_DBNAME, use_unicode=False, charset=Config.DB_CHARSET)
  23. return self.__pool.connection()
  24.  
  25. def __exit__(self, type, value, trace):
  26. """
  27. @summary: 释放连接池资源
  28. """
  29. self.cursor.close()
  30. self.conn.close()
  31. print("连接池释放conn和cursor")
  32.  
  33. def getconn(self):
  34. '''
  35. 从连接池中取出一个连接
  36. '''
  37. conn = self.__getConn()
  38. cursor = conn.cursor(pymysql.cursors.DictCursor)
  39. return cursor, conn
  40.  
  41. def close(self):
  42. '''
  43. 关闭连接归还给连接池
  44. '''
  45. self.cursor.close()
  46. self.conn.close()
  47. print("连接池释放conn和cursor")
  48.  
  49. POOL = ConnectionPool()
  50.  
  51. class MysqlHelper(object):
  52. mysql = None
  53.  
  54. def __init__(self):
  55. self.db = POOL
  56.  
  57. def __new__(cls, *args, **kwargs):
  58. if not hasattr(cls, 'inst'):
  59. cls.inst = super(MysqlHelper, cls).__new__(cls, *args, **kwargs)
  60. return cls.inst
  61.  
  62. def selectall(self, sql='', param=()):
  63. '''
  64. 查询所有
  65. '''
  66.  
  67. try:
  68. cursor, conn = self.execute(sql, param)
  69. res = cursor.fetchall()
  70. self.close(cursor, conn)
  71. return res
  72. except Exception as e:
  73. print('selectall except ', e.args)
  74. self.close(cursor, conn)
  75. return None
  76.  
  77. def selectone(self, sql='', param=()):
  78. '''
  79. 查询一条
  80. '''
  81. try:
  82. cursor, conn = self.execute(sql, param)
  83. res = cursor.fetchone()
  84. self.close(cursor, conn)
  85. return res
  86. except Exception as e:
  87. print('selectone except ', e.args)
  88. self.close(cursor, conn)
  89. return None
  90.  
  91. def insert(self, sql='', param=()):
  92. '''
  93. 增加
  94. '''
  95. try:
  96. cursor, conn = self.execute(sql, param)
  97. print('============')
  98. _id = cursor.lastrowid
  99. print('_id ', _id)
  100. conn.commit()
  101. self.close(cursor, conn)
  102. # 防止表中没有id返回0
  103. if _id == 0:
  104. return True
  105. return _id
  106. except Exception as e:
  107. print('insert except ', e.args)
  108. conn.rollback()
  109. self.close(cursor, conn)
  110. # self.conn.rollback()
  111. return 0
  112.  
  113. def insertmany(self, sql='', param=()):
  114. '''
  115. 增加多行
  116. '''
  117. cursor, conn = self.db.getconn()
  118. try:
  119. cursor.executemany(sql, param)
  120. conn.commit()
  121. self.close(cursor, conn)
  122. return True
  123. except Exception as e:
  124. print('insert many except ', e.args)
  125. conn.rollback()
  126. self.close(cursor, conn)
  127. return False
  128.  
  129. def delete(self, sql='', param=()):
  130. '''
  131. 删除
  132. '''
  133. try:
  134. cursor, conn = self.execute(sql, param)
  135. self.close(cursor, conn)
  136. return True
  137. except Exception as e:
  138. print('delete except ', e.args)
  139. conn.rollback()
  140. self.close(cursor, conn)
  141. return False
  142.  
  143. def update(self, sql='', param=()):
  144. '''
  145. 更新
  146. '''
  147. try:
  148. cursor, conn = self.execute(sql, param)
  149. self.close(cursor, conn)
  150. return True
  151. except Exception as e:
  152. print('update except ', e.args)
  153. conn.rollback()
  154. self.close(cursor, conn)
  155. return False
  156.  
  157. @classmethod
  158. def getInstance(self):
  159. if MysqlHelper.mysql == None:
  160. MysqlHelper.mysql = MysqlHelper()
  161. return MysqlHelper.mysql
  162.  
  163. # 执行命令
  164. def execute(self, sql='', param=(), autoclose=False):
  165. cursor, conn = self.db.getconn()
  166. try:
  167. if param:
  168. cursor.execute(sql, param)
  169. else:
  170. cursor.execute(sql)
  171. conn.commit()
  172. if autoclose:
  173. self.close(cursor, conn)
  174. except Exception as e:
  175. pass
  176. return cursor, conn
  177.  
  178. def executemany(self, list=[]):
  179. '''
  180. # 执行多条命令
  181. '[{"sql":"xxx","param":"xx"}....]'
  182. '''
  183. cursor, conn = self.db.getconn()
  184. try:
  185. for order in list:
  186. sql = order['sql']
  187. param = order['param']
  188. if param:
  189. cursor.execute(sql, param)
  190. else:
  191. cursor.execute(sql)
  192. conn.commit()
  193. self.close(cursor, conn)
  194. return True
  195. except Exception as e:
  196. print('execute failed========', e.args)
  197. conn.rollback()
  198. self.close(cursor, conn)
  199. return False
  200.  
  201. def close(self, cursor, conn):
  202. cursor.close()
  203. conn.close()
  204. print("PT连接池释放con和cursor")

mysqlhelper

python之数据库连接池DBUtils的更多相关文章

  1. Python的数据库连接池DBUtils

    DBUtils是Python的一个用于实现数据库连接池的模块. 此连接池有两种连接模式: 模式一:为每个线程创建一个连接,线程即使调用了close方法,也不会关闭,只是把连接重新放到连接池,供自己线程 ...

  2. Python数据库连接池---DBUtils

    Python数据库连接池DBUtils   DBUtils是Python的一个用于实现数据库连接池的模块. 此连接池有两种连接模式: 模式一:为每个线程创建一个连接,线程即使调用了close方法,也不 ...

  3. Python数据库连接池DBUtils

    Python数据库连接池DBUtils   DBUtils是Python的一个用于实现数据库连接池的模块. 此连接池有两种连接模式: 模式一:为每个线程创建一个连接,线程即使调用了close方法,也不 ...

  4. Python数据库连接池DBUtils.PooledDB

    DBUtils 是一套用于管理数据库连接池的包,为高频度高并发的数据库访问提供更好的性能,可以自动管理连接对象的创建和释放.最常用的两个外部接口是 PersistentDB 和 PooledDB,前者 ...

  5. Python 数据库连接池DButils

    常规的数据库链接存在的问题: 场景一: 缺点:每次请求反复创建数据库连接,连接数太多 import pymysql def index(): conn = pymysql.connect() curs ...

  6. Python数据库连接池DBUtils(基于pymysql模块连接数据库)

    安装 pip3 install DBUtils DBUtils是Python的一个用于实现数据库连接池的模块. 此连接池有两种连接模式: # BDUtils数据库链接池: 模式一:基于threaing ...

  7. Python数据库连接池DBUtils详解

    what's the DBUtils DBUtils 是一套用于管理数据库连接池的Python包,为高频度高并发的数据库访问提供更好的性能,可以自动管理连接对象的创建和释放.并允许对非线程安全的数据库 ...

  8. Flask中使用数据库连接池 DBUtils ——(4)

    DBUtils是Python的一个用于实现数据库连接池的模块. 此连接池有两种连接模式: 模式一:为每个线程创建一个连接,线程即使调用了close方法,也不会关闭,只是把连接重新放到连接池,供自己线程 ...

  9. Flask的数据库连接池 DBUtils

    Flask是没有ORM的操作的,如果在flask中连接数据库有两种方式 一.pymysql 二.SQLAlchemy 是python操作数据库的以一个库,能够进行orm映射官网文档 sqlchemy ...

随机推荐

  1. ProtoBuf3 C++使用篇

    protobuf 是用于结构化数据串行化的灵活.高效.自动化的解决方案.又如 XML,不过它更小.更快.也更简单.你只需要按照你想要的数据存储格式编写一个.proto,然后使用生成器生成的代码来读写这 ...

  2. 解决ubuntu系统中firefox无法播放网页版音乐播放器音乐

    Reference: https://blog.csdn.net/h736131708/article/details/80775382 因为网页版的qq音乐或者网易云音乐都把音频换成了AAC格式,这 ...

  3. 集合(从本部分开始涉及API)

    集合(从本部分开始涉及API) 集合是指一个对象容纳了多个对象,这个集合对象主要用来管理维护一系列相似的对象. 数组就是一种对象.(练习:如何编写一个数组程序,并进行遍历.) java.util.*定 ...

  4. Ubuntu 16.04 ROS环境配置

    最近新入职一家公司,是搞智能无人驾驶的,用的操作系统是Ubuntu和ros,之前没接触过ros系统,既然公司用那就必须的学习啊,话不多说先装它一个ros玩玩... 1. Ubuntu 安装 ROS K ...

  5. windows 端口被占用,并杀死进程的方法

    netstat -ano | findstr 8081 查询端口 被什么进程占用 tasklist | findstr 2184 根据进程号 查询任务名称 taskkill /f /t /im jav ...

  6. js 原型链和继承(转)

    在理解继承之前,需要知道 js 的三个东西: 什么是 JS 原型链 this 的值到底是什么 JS 的 new 到底是干什么的 1. 什么是 JS 原型链? 我们知道 JS 有对象,比如 var ob ...

  7. MTK NTP和NITZ更新时间的问题

    NITZ(Network Identity and Time Zone,网络标识和时区),是一种用于自动配置本地的时间和日期的机制,同时也通过无线网向移动设备提供运营商信息.NITZ是自从PHASE ...

  8. 微信小程序开发--富文本插件wxParse的使用

    昨天一位网友问我小程序怎么解析富文本.他尝试过把html转出小程序的组件,但是还是不成功,我说可以把内容剥离出来.但是这两种方法都是不行了.后来找到了wxParse-微信小程序富文本解析组件. 特性 ...

  9. 排序算法--希尔排序(Shell Sort)_C#程序实现

    排序算法--希尔排序(Shell Sort)_C#程序实现 排序(Sort)是计算机程序设计中的一种重要操作,也是日常生活中经常遇到的问题.例如,字典中的单词是以字母的顺序排列,否则,使用起来非常困难 ...

  10. 学习Vue 入门到实战——学习笔记(二)

    闲聊: 哈哈哈!过了好几天才更新博客啦,嘻嘻,马上过年了,大家最近是不是都开心的快飞起来了,小颖好几个朋友公司都已经放假了,可是我们公司要等到腊月29上完班才给放假,哎!心情不美气的很,用我之前大学舍 ...