最近項目用到了hibernate3.0 c3po mysql的資料層組合,開發部署非常順利,但每天早上訪問應用都拋出 Could not open Hibernate session for transaction 異常,Caused by: com.mysql.jdbc.CommunicationsException: Communications link failure due to underlying exception,但經過幾次訪問後又恢複正常,datasource配置如下:
<bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource" destroy-method="close">
<property name="driverClass">
<value>com.mysql.jdbc.Driver</value>
</property>
<property name="jdbcUrl">
<value><![CDATA[jdbc:mysql://localhost:3306/WAP2?useUnicode=true&characterEncoding=latin1]></value>
</property>
<property name="user">
<value>root</value>
</property>
<property name="password">
<value>zznode</value>
</property>
<property name="minPoolSize">
<value>3</value>
</property>
<property name="maxPoolSize">
<value>20</value>
</property>
<property name="maxStatements">
<value>50</value>
</property>
<property name="maxIdleTime">
<value>6000000</value>
</property>
</bean>
根據問題現象初步認為是資料庫連接出現問題,逐用異常作為關鍵詞google(這是我遇見不能解決的問題時慣用的方法);得到了相關資訊:1、mysql 資料庫連結預設的逾時時間長度為28800秒,2、c3p0 的幾個參數意思;
通過以上資訊的收集隱隱知道了問題所在(mysql經過28800秒也就是8個小時後關掉空閑連結,而c3p0是經過6000000妙才取消連結,就有可能出現c3po持續連線有可能已經被mysql關掉了,自然就出現了hibernate不能開啟session,並且都是第二天一早就出錯);為了驗證我的想法,決定在開發機上重現這個錯誤;首先在mysql設定檔my.ini 加上 wait_timeout=30 讓mysql經過30妙就關掉連結,重啟應用,第一次訪問成功,等待一分鐘後訪問果然出現同樣的錯誤,說明問題診斷正確,著手修改配置如下:
<property name="maxIdleTime">
<value>28000</value>
</property>
<property name="idleConnectionTestPeriod">
<value>28000</value>
</property>
配置解釋:28000<28800 使c3p0 在mysql關不串連之前關閉自己持有的連結,配置idleConnectionTestPeriod 參數使c3po每隔28000檢查已有的串連是否可用,這樣應該確保拿到的串連都是可用的,如果還不放心可以加上 testConnectionOnCheckout參數每當拿出串連的時候就檢查一下是否可以,這個可能會使mysql有一定的效能犧牲;
<property name="testConnectionOnCheckout">
<value>true</value>
</property>
通過以上修改檢測一段時間沒有出現同樣問題,問題解決!總結解決問題的三個步驟:收集相關資訊、重現問題、解決測試。