再淡spring jdbc 連線池斷開重連設定
阿新 • • 發佈:2019-02-16
先看一段錯誤日誌:
### Error querying database. Cause: com.mysql.jdbc.exceptions.jdbc4.MySQLNonTransientConnectionException: No operations allowed after connection closed. ### The error may exist in file …………] ### The error may involve ..... ### The error occurred while executing a query ### Cause: com.mysql.jdbc.exceptions.jdbc4.MySQLNonTransientConnectionException: No operations allowed after connection closed. ; SQL []; No operations allowed after connection closed.; nested exception is com.mysql.jdbc.exceptions.jdbc4.MySQLNonTransientConnectionException: No operations allowed after connection closed.
網上一搜,解決辦法一大堆,基本配置如下:
<bean id="dataSource" class="org.apache.commons.dbcp.BasicDataSource" destroy-method="close"> <property name="driverClassName" value="${jdbc.driverClassName}" /> <property name="url" value="${jdbc.url}" /> <property name="username" value="${jdbc.username}" /> <property name="password" value="${jdbc.password}" /> <!-- 佇列中的最小等待數 --> <property name="minIdle" value="${jdbc.minIdle}"></property> <!-- 佇列中的最大等待數 --> <property name="maxIdle" value="${jdbc.maxIdle}"></property> <!-- 最長等待時間,單位毫秒 --> <property name="maxWait" value="${jdbc.maxWait}"></property> <!-- 最大活躍數 --> <property name="maxActive" value="${jdbc.maxActive}"></property> <property name="initialSize" value="${jdbc.initialSize}"></property> <property name="validationQuery" value="select 1"/> <property name="testOnBorrow" value="false"/> <property name="testWhileIdle" value="true"/> <property name="testOnReturn" value="false"/> <property name="numTestsPerEvictionRun" value="${jdbc.maxActive}"/> <!-- 5 min 每5分鐘檢測空閒連線超過10分鐘的連線--> <property name="timeBetweenEvictionRunsMillis" value="300000" /> <property name="minEvictableIdleTimeMillis" value="600000" /> <property name="removeAbandoned" value="true"/> </bean>
可是,這就夠了嗎?
一開始放到自己的環境上是沒有用的,還是報錯了。
繼續努力,最後是解決了。
第一要理解連線池的各項配置(上面)
第二是要知道mysql中wait_timeout的設定
兩點結合才能確定連線池在專案中的合理正確配置。
如果wait_timeout設定成很大一個值,例如一年,那麼上面的配置很多情況下都是正確的。
如果wait_timeout設定成很小,如1分鐘,那麼上面的配置是有問題的。因為伺服器1分鐘就把空閒連線斷開了,客戶端過了5分鐘再去檢查連線情況,那有什麼意義?先前就是沒理解被誤導了,把timeBetweenEvictionRunsMillis設定了一個比較大的值,所以一直有問題。包括所說的8小時問題也是源於此(mysql資料庫預設是空閒8小時斷開)。
我的原因是mysql的wait_timeout的值設定小了,而客戶端檢測的間隔時間過大。
正確的做法是:
連線池配置中的timeBetweenEvictionRunsMillis和
minEvictableIdleTimeMillis的
時間小於或者等於mysql資料庫中wait_timeout的時間。