Django數(shù)據(jù)庫連接丟失問題的解決方法
問題
在Django中使用mysql偶爾會(huì)出現(xiàn)數(shù)據(jù)庫連接丟失的情況,錯(cuò)誤通常有如下兩種
OperationalError: (2006, 'MySQL server has gone away')
OperationalError: (2013, 'Lost connection to MySQL server during query')
查詢mysql全局變量SHOW GLOBAL VARIABLES;可以看到wait_timeout,此變量表示連接空閑時(shí)間。如果客戶端使用一個(gè)連接查詢多次數(shù)據(jù)庫,如果連續(xù)查詢則沒有問題,如果查詢幾次后停頓超過wait_timeout后再次查詢就會(huì)出現(xiàn)數(shù)據(jù)庫連接丟失。
復(fù)現(xiàn)
下面用Django復(fù)現(xiàn)下次問題:
將mysql的wait_timeout設(shè)置為10秒,然后進(jìn)入django shell模擬查詢(以下錯(cuò)誤信息只保留了部分)
In[1]:import time
In[2]:from django.contrib.auth.models import User
In[3]:list(User.objects.filter(id=1))
Out[3]:[<User: admin>]
In[4]:time.sleep(15) # 模擬比較慢的代碼(其中沒有查詢數(shù)據(jù)庫的代碼),或者空閑什么都不操作一段時(shí)間,此時(shí)間要比`wait_timeout`大一些
list(User.objects.filter(id=1))
Traceback (most recent call last):File "<ipython-input-4-3574ae8220ee>", line 1, in <module>
list(User.objects.filter(id=1))File "/usr/lib/python3.6/site-packages/pymysql/connections.py", line 1037, in _read_bytes
CR.CR_SERVER_LOST, "Lost connection to MySQL server during query")
django.db.utils.OperationalError: (2013, 'Lost connection to MySQL server during query')
尋求
那么以上問題就基本說明了是空閑時(shí)間過長導(dǎo)致的錯(cuò)誤。
django為了減少不必要的數(shù)據(jù)庫連接、關(guān)閉,復(fù)用了數(shù)據(jù)庫連接,當(dāng)開始一個(gè)請求后建立一個(gè)連接池存放連接,之后此次請求都復(fù)用一個(gè)連接。那猜測就是django保存連接的比wait_timeout長了,如果保存時(shí)間短一些就可以重新建立連接避免此錯(cuò)誤了。 沒錯(cuò),官方文檔也已經(jīng)說明了此問題,設(shè)置數(shù)據(jù)庫 CONN_MAX_AGE參數(shù),示例:
DATABASES = {
"default": {
'ENGINE': 'django.db.backends.mysql',
'NAME': '',
'USER': '',
'PASSWORD': '',
'HOST': '',
'CONN_MAX_AGE': 9 # 比wait_timeout小一些
}
}
當(dāng)我們測試后卻發(fā)現(xiàn),事情并非想想中那么簡單。為何錯(cuò)誤依舊出現(xiàn)?這一切的背后, 是人性的扭曲還是道德的淪喪?敬請收看下節(jié)《突破》。
突破
對django源碼中CONN_MAX_AGE進(jìn)行了一番搜索,順藤摸瓜發(fā)現(xiàn)了django關(guān)閉失效連接的方法django.db.close_old_connections():
# Register an event to reset transaction state and close connections past # their lifetime. def close_old_connections(**kwargs): for conn in connections.all(): conn.close_if_unusable_or_obsolete() signals.request_started.connect(close_old_connections) signals.request_finished.connect(close_old_connections)
重點(diǎn)在最后兩行,通過signal實(shí)現(xiàn)特定事件時(shí)執(zhí)行此方法,兩個(gè)特定事件顧名思義是請求開始和請求結(jié)束。而我們報(bào)錯(cuò)的是在一次請求中,所以此法通常無效,僅僅是實(shí)現(xiàn)每個(gè)請求關(guān)閉并重新建立連接。
解決
復(fù)現(xiàn)問題的django shell不要關(guān)閉,繼續(xù)執(zhí)行如下代碼:
In[5]:from django.db import close_old_connections
In[6]:close_old_connections()
In[7]:list(User.objects.filter(id=1))
Out[7]: [<User: admin>]
調(diào)用django.db.close_old_connections后再次查詢就沒有錯(cuò)誤了。 那么我們要避免此錯(cuò)誤就要執(zhí)行每個(gè)數(shù)據(jù)庫查詢前調(diào)用django.db.close_old_connections方法。
一般情況不會(huì)出現(xiàn)此類問題,因?yàn)橐粋€(gè)請求中不間斷進(jìn)行數(shù)據(jù)庫查詢,無需每個(gè)請求調(diào)用此方法,杞人憂天。
有時(shí)候一個(gè)請求中數(shù)據(jù)量較大,會(huì)查詢數(shù)據(jù)庫后進(jìn)行一段時(shí)間其他(不涉及數(shù)據(jù)庫)處理,比如先查詢一些數(shù)據(jù),然后將數(shù)據(jù)處理、生成excel、保存文件并生成url。已知此過長需要非常長時(shí)間,那么最終url保存數(shù)據(jù)庫就最好先調(diào)用django.db.close_old_connections防止連接丟失
題外話 實(shí)際上②所述情況最好從根本上解決處理慢的問題,也可以換作異步處理,從根本上解決問題。
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Python機(jī)器學(xué)習(xí)多層感知機(jī)原理解析
最簡單的深度網(wǎng)絡(luò)稱為多層感知機(jī),它們由多層神經(jīng)元組成,每一層都與下面一層(從中接收輸入)和上面一層(反過來影響當(dāng)前層的神經(jīng)元)完全相連2021-10-10
Python pandas進(jìn)行數(shù)據(jù)預(yù)處理的實(shí)現(xiàn)
本案例通過使用pandas庫對電子商務(wù)客戶數(shù)據(jù)進(jìn)行數(shù)據(jù)預(yù)處理,包括數(shù)據(jù)導(dǎo)入、查看、缺失值處理等處理,具有一定的參考價(jià)值,感興趣的可以了解一下2025-01-01
Python常用數(shù)據(jù)類型之間的轉(zhuǎn)換總結(jié)
在本篇文章里小編給大家整理的是關(guān)于Python中常用數(shù)據(jù)類型之間的轉(zhuǎn)換相關(guān)知識點(diǎn),有需要的朋友們可以學(xué)習(xí)下2019-09-09
python 實(shí)現(xiàn)判斷ip連通性的方法總結(jié)
下面小編就為大家分享一篇python 實(shí)現(xiàn)判斷ip連通性的方法總結(jié),具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧2018-04-04
Python和Pycharm 環(huán)境部署詳細(xì)步驟
Python環(huán)境搭建過程很多朋友都操作過,本次我們將向大家介紹Python和Pycharm 環(huán)境部署的流程,文章通過圖文的形式給大家展示一目了然一看就懂,需要的朋友參考下吧2021-06-06
Python數(shù)據(jù)清洗之利用pandas篩選數(shù)據(jù)詳解
這篇文章主要介紹了Python數(shù)據(jù)清洗之利用pandas篩選數(shù)據(jù)詳解,Pandas是一個(gè)用于數(shù)據(jù)分析和處理的Python庫,它提供了高效的數(shù)據(jù)結(jié)構(gòu)和數(shù)據(jù)分析工具,使得數(shù)據(jù)的清洗、轉(zhuǎn)換、分析和可視化變得更加容易和靈活,需要的朋友可以參考下2023-08-08
Django 路由層URLconf的實(shí)現(xiàn)
這篇文章主要介紹了Django 路由層URLconf的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-12-12

