时间:2021-05-22
工作中纯服务端的项目用到了线程池和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 数据库连接模块解析及简单长连接改造方法就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持。
声明:本页内容来源网络,仅供用户参考;我单位不保证亦不表示资料全面及准确无误,也不保证亦不表示这些资料为最新信息,如因任何原因,本网内容或者用户因倚赖本网内容造成任何损失或损害,我单位将不会负任何法律责任。如涉及版权问题,请提交至online#300.cn邮箱联系删除。
Django项目默认使用sqlite数据库,但是我想用mysql数据库,应该如何配置呢。Django连接mysql数据库的操作,是通过根模块的配置实现的,在项目
简介Django数据库连接超过wait_timeout导致连接丢失时自动重新连接数据库https://github.com/zhanghaofe...(本地下载
前言我们在django-rest-framework解析请求参数文章中完成了接口文档到参数解析,一个完整的流程中还缺少对数据库的操作.本篇内容为django连接
本文实例讲述了C#创建数据库及附加数据库的操作方法。分享给大家供大家参考,具体如下://////附加数据库方法//////连接数据库字符串,连接master系统
本篇博文主要介绍Python连接各种数据库的方法及简单使用包括关系数据库:sqlite,mysql,mssql非关系数据库:MongoDB,Redis代码写的比