SqlDataReader的关闭问题,报错:“阅读器关闭时尝试调用 Read 无效”

最近学习了一下ADO.NET,我想封装一个获取SqlDataReader的方法供别处调用,

但是遇到了一些问题,最后找到了解决方法。
封装的时候如果用using{}将连接及执行等过程括起来,这样可以保证最后关闭连

接,但是调用时却会出错。

最初我封装的代码是这样的:

[csharp]  view plain copy
  1. public static SqlDataReader GetDataReader(string sql, params SqlParameter[] parameters)  
  2.     {  
  3.         using (SqlConnection conn = new SqlConnection(connstr))  
  4.         {  
  5.             SqlCommand cmd = new SqlCommand(sql, conn);  
  6.             foreach (SqlParameter parameter in parameters)  
  7.             {  
  8.                 cmd.Parameters.Add(parameter);  
  9.             }  
  10.   
  11.             try  
  12.             {  
  13.                 conn.Open();  
  14.                 SqlDataReader dr = cmd.ExecuteReader();  
  15.                 return dr;  
  16.             }  
  17.             catch (Exception e)  
  18.             {  
  19.                 throw new Exception("执行任务失败:" + e.Message + "   "+ sql);  
  20.             }  
  21.         }  
  22.     }  

用这样的方法进行封装时,调用GetDataReader方法报错:“阅读器关闭时尝试调用 Read 无效”。这是因为出了using{}的作用域之后,连接自动关闭,而Reader与DataSet不同,DataSet已经将数据保存在本地内存中,而Reader并没有将数据保存在本地内存中,可以理解成只是指向了数据,连接关闭后用reader读取数据当然无法实现。
于是我参考许多资料后,对我的代码进行了修改,并且验证了网上几个说法不一
的问题。

修改后的代码如下:

[csharp]  view plain copy
  1. public static SqlDataReader GetDataReader(string sql, params SqlParameter[] parameters)  
  2.     {  
  3.         SqlConnection conn = new SqlConnection(connstr);  
  4.         SqlCommand cmd = new SqlCommand(sql, conn);  
  5.         foreach (SqlParameter parameter in parameters)  
  6.         {  
  7.             cmd.Parameters.Add(parameter);  
  8.         }  
  9.   
  10.         try  
  11.         {  
  12.             conn.Open();  
  13.             SqlDataReader dr = cmd.ExecuteReader(CommandBehavior.CloseConnection);//关闭SqlDataReader 会自动关闭Sqlconnection  
  14.             return dr;  
  15.         }  
  16.         catch (Exception e)  
  17.         {  
  18.             throw new Exception("执行任务失败:" + e.Message + "   " + sql);  
  19.         }      
  20.           
  21.     }  


这样封装好之后,我在一个网页的后台代码中进行了调用:

[csharp]  view plain copy
  1. protected void TextBoxName_TextChanged(object sender, EventArgs e)  
  2.     {  
  3.         string UserN = TextBoxName.Text;  
  4.         string strsql = "select * from UserInfo where UI01=@UserName";  
  5.   
  6.         SqlDataReader dr = DBN.GetDataReader(strsql, new SqlParameter("UserName", UserN));  
  7.         if (dr.Read())  
  8.         {  
  9.             Response.Write("<script language='javascript'>alert('该用户名已存在');</script>");  
  10.             TextBoxName.Text = null;  
  11.         }  
  12.         dr.Close();//采用了close进行手动关闭。  
  13.          Response.Write(dr.IsClosed.ToString() + "<br/>");//查看SqlDataReader的状态  
  14.     }  

这样就没有在报错:“阅读器关闭时尝试调用 Read 无效”了。这里需要说明的是,在数据读取完毕之后,不再需要SqlDataReader时,必须将其进行手动关闭。因为cmd.ExecuteReader(CommandBehavior.CloseConnection)虽然可以实现关闭SqlDataReader后自动关闭连接,但是SqlDataReader必须手动关闭,上面的代码中如果去掉dr.Close(),则Response.Write(dr.IsClosed.ToString()+"<br/>")输出为False,也就是说指定的 SqlDataReader 实例没有关闭。
在MSDN中也有描述:在使用 SqlDataReader 时,关联的 SqlConnection 正忙于
为 SqlDataReader 服务,对 SqlConnection 无法执行任何其他操作,只能将其关闭。除非调用 SqlDataReader 的 Close 方法,否则会一直处于此状态。
由此看来,网上有人说不用手动关闭SqlDataReader的说法是不对的。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值