最近在使用C#代码连接Oracle数据库,分为两部分,WCF的客户端与服务端。程序启动与运行都没有问题,部署到服务器上后,运行也没有问题。但是第二天再访问的时候,就会抛出下边所示的异常。这是怎么回事?
- Oracle.DataAccess.Client.OracleException ORA-03135: connection lost contact
- Process ID: 22574
- Session ID: 799 Serial number: 43225 Oracle.DataAccess.Client.OracleException.HandleErrorHelper(Int32 errCode, OracleConnection conn, IntPtr opsErrCtx, OpoSqlValCtx* pOpoSqlValCtx, Object src, String procedure, Boolean bCheck)
- at Oracle.DataAccess.Client.OracleCommand.ExecuteReader(Boolean requery, Boolean fillRequest, CommandBehavior behavior)
- at Oracle.DataAccess.Client.OracleDataAdapter.Fill(DataSet dataSet, Int32 startRecord, Int32 maxRecords, String srcTable, IDbCommand command, CommandBehavior behavior)
- at System.Data.Common.DbDataAdapter.Fill(DataSet dataSet, String srcTable)
- public DataSet ExcuteDataSetForOralce(string sql, CommandType type, List<OracleParameter> paras)
- {
- DataSet ds = new DataSet();
- OracleCommand cmd = null;
- try
- {
- using (OracleConnection conn = new OracleConnection(ConnectionString))
- {
- using (cmd = new OracleCommand(sql, conn))
- {
- if (paras != null)
- {
- foreach (OracleParameter p in paras)
- {
- cmd.Parameters.Add(p);
- }
- }
- cmd.CommandType = type;
- OracleDataAdapter oda = new OracleDataAdapter();
- oda.SelectCommand = cmd;
- conn.Open();
- oda.Fill(ds, "tempTable");
- conn.Close();
- }
- }
- }
- catch (Exception ex)
- {
- log.Error("error", ex, LogType.Error);
- }
- return ds;
- }
通过多方查找,发现这并不是代码的问题。而且Provider与数据库连接的问题。实际上,我们new一个连接的时候,是从数据库连接池里边拿到的连接,这些连接放在.NET Provider里边。Conn.Close()并不是关闭了连接,而是把连接还给了连接池。
现在的问题是Oracle Provider的问题,而不是代码的问题。你的程序很久没有跟数据库交互了,数据库服务器端就把连接的Open状态关了,但是Provider没有及时处理,依然给了C#代码用,所以就抛出了这个异常。
解决办法就是,不使用连接池,每次直接连数据库,虽然不会出现这个问题,但是会有性能上的损失。
在app.config的配置文件中,在DataSource=((Description...)这个字符串中,加入不使用连接池。
Data Source =( DESCRIPTION =( ADDRESS_LIST =( ADDRESS =( PROTOCOL = TCP )( HOST = xxx )( PORT = 1521 )))(CONNECT_DATA =( SERVICE_NAME = xxx )(SERVER = DEDICATED)));User ID = xxx ; password = xxxxx ; Pooling = false ;