SqlDataReader 关闭问题:阅读器关闭时尝试调用 Read 无效

 

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

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

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

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

 

public static SqlDataReader GetDataReader(string sql, params SqlParameter[] parameters)
    {
        using (SqlConnection conn = new SqlConnection(connstr))
        {
            SqlCommand cmd = new SqlCommand(sql, conn);
            foreach (SqlParameter parameter in parameters)
            {
                cmd.Parameters.Add(parameter);
            }

            try
            {
                conn.Open();
                SqlDataReader dr = cmd.ExecuteReader();
                return dr;
            }
            catch (Exception e)
            {
                throw new Exception("执行任务失败:" + e.Message + "   "+ sql);
            }
        }
    }


 

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

修改后的代码如下:

protected void TextBoxName_TextChanged(object sender, EventArgs e)
    {
        string UserN = TextBoxName.Text;
        string strsql = "select * from UserInfo where UI01=@UserName";

        SqlDataReader dr = DBN.GetDataReader(strsql, new SqlParameter("UserName", UserN));
        if (dr.Read())
        {
            Response.Write("");
            TextBoxName.Text = null;
        }
        dr.Close();//采用了close进行手动关闭。
         Response.Write(dr.IsClosed.ToString() + "
");//查看SqlDataReader的状态 }


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

protected void TextBoxName_TextChanged(object sender, EventArgs e)
    {
        string UserN = TextBoxName.Text;
        string strsql = "select * from UserInfo where UI01=@UserName";

        SqlDataReader dr = DBN.GetDataReader(strsql, new SqlParameter("UserName", UserN));
        if (dr.Read())
        {
            Response.Write("");
            TextBoxName.Text = null;
        }
        dr.Close();//采用了close进行手动关闭。
         Response.Write(dr.IsClosed.ToString() + "
");//查看SqlDataReader的状态 }


 

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


 

 

你可能感兴趣的:(.Net,dataset,parameters,exception,string,javascript,object)