django 数据库连接模块解析及简单长连接改造方法


Posted in Python onAugust 29, 2019

工作中纯服务端的项目用到了线程池和django的ORM部分。django 的数据库连接在每一个线程中开启一份,并在查询完毕后自动关闭连接。

线程池处理任务时,正常使用的连接中不会被关闭,但由于数据库端有最长连接时间的限制(默认为8小时),在超时后会发生InterfaceError: (0, '')(连接关闭后使用连接/游标)或Error(2006, 'MySQL server has gone away')(mysql 服务器主动关闭连接)这类错误,所以一般会在每个任务线程中调用django.db.connection.close()进行关闭操作。

但对于频繁进行数据库连接并操作数据库的业务,反复创建连接并不是好的选择,这种场景下可以考虑将连接改造为长连接。

1. django 代码的阅读笔记

django.db.__init__.py 
#对象:
connections = ConnectionHandler()
connection = DefaultConnectionProxy()
# 函数
# 重置查询记录缓存
def reset_queries(**kwargs):
 pass
# 关闭不可用或超时(如果有设置 CONN_MAX_AGE)连接
def close_old_connections(**kwargs):
 pass
# 信号
# 在请求开始或完成时自动调用相应处理函数
signals.request_started.connect(reset_queries)
signals.request_started.connect(close_old_connections)
signals.request_finished.connect(close_old_connections)

重点是connections和connection两个实例

connections 是 ConnectionHandler类

connections.all()会给出一个列表,里面的元素为DatabaseWrapper类

ConnectionHandler内置对象及连接管理:

def __init__():
 self._connections = local()

# 连接包装类里的连接是根据配置情况使用相应的连接
def __getitem__(self, alias):
 '''略'''
 db = self.databases[alias]
 backend = load_backend(db['ENGINE'])
 conn = backend.DatabaseWrapper(db, alias)
 setattr(self._connections, alias, conn)

# 返回所管理的数据库连接
# 管理方式:分数据库,线程管理连接 
def all(self):
 return [self[alias] for alias in self]

# 关闭所有数据库连接
def close_all(self):
 for alias in self:
  try:
   connection = getattr(self._connections, alias)
  except AttributeError:
   continue
  connection.close()

threading.local 是一个全局变量,local的属性是非线程共享的,也就是在每一个线程中都会有单独一个数据库连接实例创建,因为代理及包装的原因,该连接实例为对应backend里的连接(比如,pymysql.connections.Connection)。

在线程池的情况下,close_old_connections方法是不能将线程中的数据库连接关闭的。

connection是DefaultConnectionProxy类的实例,实际是DatabaseWrapper的实例 
(使用了pymysql库:import pymysql; pymysql.install_as_MySQLdb) 
DefaultConnectionProxy?>DatabaseWrapper?>pymysql.connections.Connection(根据connections的处理调用相应的数据库连接包) 
connection有几个关键方法和属性

connection.connection = '被包装的pymysql.connections.Connection实例`
connection.close_at = None if max_age is None else time.time() + max_age # 设置的连接关闭时间

connection.connect()# 获取连接
connection.cursor() # 获取游标
connection.close()# 关闭连接

2. 将数据库连接改造为长连接

max_age(CONN_MAX_AGE) 是可以在settings里面配置的。

由于多个服务共用一套配置, 所以考虑直接在程序里修改

全局变量

max_age = 7 * 3600

在线程内开始时做下判断:

if not db.connection.connection or db.connection.close_at < time.time():
 db.connection.close()
 db.connection.connect()
 db.connection.close_at = time.time() + max_age
 print "A new conn creates !"
else:
 print "Still old conn!"

这样每个线程池中的线程会循环执行任务并只使用同一个连接,并可以控制在自己需要的连接时长后更换连接。

针对线程池的情况,close_old_connections基本没啥用处, 可以跳过该处理

django.db.close_old_connections = lambda **kwargs : None

以上这篇django 数据库连接模块解析及简单长连接改造方法就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持三水点靠木。

Python 相关文章推荐
使用python调用浏览器并打开一个网址的例子
Jun 05 Python
Python实现在matplotlib中两个坐标轴之间画一条直线光标的方法
May 20 Python
Python的Django框架中的表单处理示例
Jul 17 Python
python魔法方法-属性转换和类的表示详解
Jul 22 Python
CentOS7.3编译安装Python3.6.2的方法
Jan 22 Python
Python生成一个迭代器的实操方法
Jun 18 Python
python系列 文件操作的代码
Oct 06 Python
详解Python list和numpy array的存储和读取方法
Nov 06 Python
TensorFlow2.0:张量的合并与分割实例
Jan 19 Python
flask框架自定义url转换器操作详解
Jan 25 Python
python交互模式基础知识点学习
Jun 18 Python
Python利用matplotlib绘制折线图的新手教程
Nov 05 Python
解决Django连接db遇到的问题
Aug 29 #Python
Python pandas实现excel工作表合并功能详解
Aug 29 #Python
python openCV获取人脸部分并存储功能
Aug 28 #Python
python lambda表达式在sort函数中的使用详解
Aug 28 #Python
python实现微信小程序用户登录、模板推送
Aug 28 #Python
使用Python脚本zabbix自定义key监控oracle连接状态
Aug 28 #Python
django删除表重建的实现方法
Aug 28 #Python
You might like
php和js交互一例-PHP教程,PHP应用
2007/01/03 PHP
PHP仿博客园 个人博客(1) 数据库与界面设计
2013/07/05 PHP
Zend studio文件注释模板设置方法
2013/09/29 PHP
Yii使用Captcha验证码的方法
2015/12/28 PHP
浅谈PHP中的数据传输CURL
2016/09/06 PHP
JavaScript中的其他对象
2008/01/16 Javascript
js获取height和width的方法说明
2013/01/06 Javascript
Extjs改变树节点的勾选状态点击按钮将复选框去掉
2013/11/14 Javascript
js对table的td进行相同内容合并示例详解
2013/12/27 Javascript
javascript中数组array及string的方法总结
2014/11/28 Javascript
JavaScript如何实现组合列表框中元素移动效果
2016/03/01 Javascript
JS数组去掉重复数据只保留一条的实现代码
2016/08/11 Javascript
Vue.js使用$.ajax和vue-resource实现OAuth的注册、登录、注销和API调用
2017/05/10 Javascript
Vue引用第三方datepicker插件无法监听datepicker输入框的值的解决
2018/01/27 Javascript
JavaScript中Object基础内部方法图
2018/02/05 Javascript
JS实现可视化文件上传
2018/09/08 Javascript
浅谈angular2子组件的事件传递(任意组件事件传递)
2018/09/30 Javascript
react脚手架如何配置less和ant按需加载的方法步骤
2018/11/28 Javascript
vue实现todolist功能、todolist组件拆分及todolist的删除功能
2019/04/11 Javascript
Vue+element+cookie记住密码功能的简单实现方法
2020/09/20 Javascript
微信小程序实现拼图小游戏
2020/10/22 Javascript
[01:35]2014DOTA2西雅图邀请赛 专访狐狸妈青春献给刀塔
2014/07/08 DOTA
python中使用正则表达式的连接符示例代码
2017/10/10 Python
Python之list对应元素求和的方法
2018/06/28 Python
Python函数的默认参数设计示例详解
2019/12/01 Python
什么是Python变量作用域
2020/06/03 Python
水果花束:Fruit Bouquets
2017/12/20 全球购物
秋游活动策划方案
2014/02/16 职场文书
大学生秋游活动方案
2014/02/17 职场文书
民族学专业求职信
2014/07/28 职场文书
四风自我剖析材料思想汇报
2014/10/01 职场文书
导游词之河北邯郸
2019/09/12 职场文书
Java基础之this关键字的使用
2021/06/30 Java/Android
Win11怎样将锁屏账户头像图片改成动画视频
2021/11/21 数码科技
Python数据可视化之Seaborn的安装及使用
2022/04/19 Python
Oracle锁表解决方法的详细记录
2022/06/05 Oracle