解決MySQL server has gone away錯誤的解決方案
PHP中MySQL server has gone away問題
一.背景
之前在Codeigniter裡面寫過類似console命令列的指令碼. 指令碼里存在sleep語句時間比較久, 導致出現一個現象就是sleep之前的SQL都是操作成功的,但是sleep之後,再執行SQL操作竟然報錯: MySQL server has gone away. 也就是mysql的這個連線失效. 後來分析才知道, MySQL中存在2個重要的配置引數:
interactive_timeout
wait_timeout
這2個引數的單位都是秒(s). 預設是8小時(28800). interactive_timeout從單詞上看指的是互動超時時間. mysql的連線方式一般分為2種, 一種稱之為"互動式", 一種稱為"非互動式". 一般常見的使用mysql -u root xxx之類的或者主從複製的連線為"互動式連線", 使用如Java的JDBC、PHP的PDO驅動連線的方式一般是"非互動式連線". 然而interactive_timeout如果未修改的情況下,這個值是一直不會變的,但是wait_timeout在不同連線方式下,值是不一樣的.
wait_timeout在"互動式連線"下, 其值是interactive_timeout的值. 如果在"非互動式連線"情況下, 則wait_timeout的值是原來mysql.cnf中配置的原始值.
最終起作用的只是wait_timeout的值.這配置項限定了處於sleep狀態(通過 show processlist檢視當前連線數情況)的連線,如果這個連線sleep休眠時間超過wait_timeout的值,則這個連線被斷掉或者說被清理掉.
二.WAIT_TIMEOUT分析
1.首先檢視mysql.conf配置
首先我們配置了interactive_timeout=10 wait_timeout=5, 此時通過mysql客戶端(互動式連線)檢視這2個配置項的值: show variables like ‘%timeout%';
2.互動式連線
客戶端的結果: wait_timeout竟然不是我們msyql.conf配置的10s, 而是 5s.
那我們再來看看PHP連線MySQL(非互動式連線),執行相同的語句,得到什麼結構:
3.非互動式連線
此時wait_timeout是我們原來在mysql.cnf配置的值了.
綜上所述: wait_timeout這個值,在不同的"連線模式"下面,拿到的值是不一樣的.
三.GONE AWAY原因分析
結合上面的情況,我們就知道了。 一開始某些SQL執行成功,但是後面的SQL執行失敗報錯gone away,大部分原因就是這個連線被閒置超過了wait_timeout,mysql伺服器單方面斷掉了這個連線。但是客戶端程式碼,還是在用這個連線變數,以為連線還是ok的(其實mysql server端已經斷開了,只是我們以為這個連線還有效),去執行SQL必然報錯.
那麼我們怎麼解決這個情況呢?
1.可以適當調整wait_timeout的值, 調大一點,這樣不容易觸發這個gone away情況.但是弊端就是,sleep的長連線不被清理,資源白白浪費了.
2.通過try-cach如果丟擲gone way msyql的連線問題, 先把之前的db呼叫close().在重新獲取db連線open,然後再執行之前的程式碼. 不過程式碼看起來感覺很蛋疼.虛擬碼:
1
2
3
4
5
6
7
8
9
10
11
12
13 |
$db = db();
try {
fun1$db); // 一開始執行成功
sleep(3600*10) // 假設sleep了10個小時
fun2($db); // 10小時之後 由於連線已經被mysql幹掉 導致報錯 gone away
}catch(Exception $e) {
// 報錯後 我們把無效的連線close 在 open新的連線
$db->close();
$db->open();
// 再拿到新的連線執行
// fun2($db)
} |
3.如果是使用類似swoole或者easyswoole框架, 建議使用mysql pool連線池的形式.並且一般連線池都有關於心跳檢查ping、連線存活檢測間隔時間設定、最大閒置連線數等等設定, 只要配置一次就好了。 例如可以配置測活連線間隔時間短一點,來保證連線不會被msyql伺服器幹掉.例如 easyswoole配置:
例如之前我設定wait_timeout=10, 但是如果我沒修改這個easyswoole的mysql連線池測活間隔時間變小, 同樣會出現gone way的情況. 第一次訪問介面成功返回SQL執行結果,但是超過10s以後我再次訪問介面,報錯mysql has gone away。修改setIntervalCheckTime()之後,就不會出現這個問題了. 我們通過mysql的show processlist;檢視連線數情況:
這些都是easyswoole幫我們維護的連線數. 當sleep超過3秒時, 由於檢查時間是3秒存活, 連線池幫我們保活檢查, sleep的時間又從0開始計算.
下面是其他網友的補充
進入MySQL
cmd
mysql -u使用者名稱 -p密碼
在我們使用mysql匯入大檔案sql時可能會報MySQL server has gone away錯誤,該問題是max_allowed_packet配置的預設值設定太小,只需要相應調大該項的值之後再次匯入便能成功。該項的作用是限制mysql服務端接收到的包的大小,因此如果匯入的檔案過大則可能會超過該項設定的值從而導致匯入不成功!下面我們來看一下如何檢視以及設定該項的值。
檢視 max_allowed_packet 的值
show global variables like 'max_allowed_packet';
+--------------------+---------+
| Variable_name | Value |
+--------------------+---------+
| max_allowed_packet | 4194304 |
+--------------------+---------+
可以看到預設情況下該項的大小隻有4M,接下來將該值設定成150M(1024*1024*150)
set global max_allowed_packet=157286400;
此時再檢視大小
show global variables like 'max_allowed_packet';
通過調大該值,一般來說再次匯入資料量大的sql應該就能成功了,如果任然報錯,則繼續再調大一些就行,請注意通過在命令列中進行設定只對當前有效,重啟mysql服務之後則恢復預設值,但可以通過修改配置檔案(可以在配置檔案my.cnf中新增max_allowed_packet=150M即可)來達到永久有效的目的,可其實我們並不是經常有這種大量資料的匯入操作,所以個人覺得通過命令列使得當前配置生效即可,沒有必要修改配置檔案。
來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/69957453/viewspace-2749290/,如需轉載,請註明出處,否則將追究法律責任。
相關文章
- MySQL server has gone away 解決方案MySqlServerGo
- MySQL server has gone away 問題的解決方法MySqlServerGo
- MySQL server has gone awayMySqlServerGo
- 問題MySQL server has gone awayMySqlServerGo
- MySQL server has gone away錯誤的一些常見原因分析MySqlServerGo
- MySQL Insert資料量過大導致報錯 MySQL server has gone awayMySqlServerGo
- Thinkphp mysql 資料庫斷線重連 MySQL server has gone awayPHPMySql資料庫ServerGo
- HTTP 錯誤 500.21 - Internal Server Error 解決方案HTTPServerError
- SQL Server 錯誤程式碼:17058 。解決方案SQLServer
- MySQL插入資料1366錯誤解決方案MySql
- MySQL錯誤Incorrect file format解決方案薦MySqlORM
- mysql 匯入 mysqldump 備份得資料保 gone away錯誤MySqlGo
- MySQL連線錯誤(10048)的解決方案MySql
- [轉]MySql錯誤程式碼1045的解決方案MySql
- latex 錯誤以及解決方案
- 【MySQL】解決mysql的 1594 錯誤MySql
- mysql與php錯誤解決MySqlPHP
- mysql錯誤解決總結MySql
- MySQL資料庫常見錯誤及解決方案MySql資料庫
- Linux下錯誤解決方案Linux
- 解決ntp的錯誤 no server suitable for synchronization foundServerUI
- This function has none of DETEMINISTIC,NO SQL錯誤解決辦法FunctionNoneSQL
- Myeclipse 錯誤An internal error has occurred 解決辦法EclipseError
- SQL SERVER Msg 7357錯誤解決SQLServer
- MySQL資料庫錯誤server_errno=2013的解決MySql資料庫Server
- Can't Connect to MySQL Server on IP Address (10061) 錯誤的解決方案MySqlServer
- Ubuntu Cannot allocate memory 錯誤解決方案Ubuntu
- ORA-28000錯誤解決方案
- ORA-07445: 錯誤解決方案
- ORA-00054錯誤解決方案(摘)
- Server Application Error報錯資訊的解決方案ServerAPPError
- MYSQL中 TYPE=MyISAM 錯誤的解決方法MySql
- vmware server啟動時503錯誤解決Server
- centos Vmware虛擬機器 Oh no,something has gone wrong! 解決方法CentOS虛擬機Go
- HTTP 錯誤 500.19- Internal Server Error 錯誤解決方法HTTPServerError
- ORA-00845的錯誤與解決方案
- SVN 遇到的一些錯誤解決方案
- PbootCMS中常見的錯誤提示及其解決方案boot