在 ExcuteReader 中使用输出参数的问题

问题

今天一个查询需要通过 ExcuteReader  返回结果集,同时又想输出参数,刚开始的时候一直得不到输出参数的值,以为存储过程出错,但是在查询分析器里面测试是正确的,而且输出参数确实已经赋值。
更加让人百思不得其解的是,对出输出强制类型转换丢出异常之后,确又可以得到了,难道是ado.net 的bug,想象页不可能啊,这么常用的API,不可能出这种错吧,我的代码类似一下场景:

 try {
            using (SqlDataReader dr = cmd.ExecuteReader(CommandBehavior.CloseConnection)) {
                int val = (int)cmd.Parameters[1].Value; // 现在还是空值
                // more
            }
        }
        catch(Exception exp) {
            throw new ApplicationException("输出参数值:" + cmd.Parameters[1].Value, exp); // 现在可以得到输出值了
        }

真是郁闷惨,足足调试跟踪一个小时

终于还是在MSDN中找到了答案:
当您将 Command 对象用于存储过程时,可以将 Command 对象的 CommandType 属性设置为 StoredProcedure。当 CommandType 为 StoredProcedure 时,可以使用 Command 的 Parameters 属性来访问输入及输出参数和返回值。无论调用哪一个 Execute 方法,都可以访问 Parameters 属性。但是,当调用 ExecuteReader 时,在 DataReader 关闭之前,将无法访问返回值和输出参数。

ref:http://msdn2.microsoft.com/zh-CN/library/tyy0sz6b.aspx

原来如此啊,觉得又被MS忽悠了,想来,谁叫自己学艺不经啊,而且早改查文档

回到自己的代码环境,还是可以解释的。
因为当catch到Expception 的时候已经跳出 using 范围了,DataReader已经自动被关闭了,自然可以得到 输出参数的值。
当然,如果把try catch 放到 using中还是得不到的,因为还在 using范围内,DataReader并没有被关闭。

另外,MSDN中说只有关闭DataReader才可以访问,其实不然。
经过测试,可以总结如下:
1。对于ExecuteReader而言,Output parm 和 returnvalue 作为一个结果集返回DataReader,并且改结果集总是在最后一个。
2。根据1,当有结果集时,要访问输出参数和返回值,需要调用 NextResult 到输出参数和返回值对应的结果集位置
3。根据1 ,当Execute没有返回结果集时,就可以直接访问(注意,无需调用Read())
4。特别注意的对于返回多个结果集的,需要调用多次NextResult;如果结果集个数又是动态的,那么当nextResult()的返回值为 false就是了。
5。即使对ExecuteReader指定了选项CommandBehavior.SingleResult(返回单个结果集,其实是返回批处理的第一个结果集),输出参数也作为一个结果集返回。
6。关闭DataReader(Close()),会填充输出参数,因此可以访问。
7。由于DataReader是只读向前的,因此,即使是通过在关闭DataReader之前通过NextResult方问到了输出参数,之前的结果集都无法在访问了(某些情况,可能想通过输出参数来动态控制结果集的访问)。
8。为解决6中的问题,可以将不使用输出参数,直接将输出参数作为第一个结果返回(SELECT @parmname)

以上只是,自己的总结,希望没有出入,对初学者也许有帮助。

ref:
http://www.bigcircleboy.net/583a194f-2c2c-4662-9036-4e2f0eb262396084313157728108.html

时间: 2024-11-06 16:04:17

在 ExcuteReader 中使用输出参数的问题的相关文章

关于C#取SQL Server存储过程中的结果集和输出参数的问题

问题描述 今天在写一个例子,用C#取存储过程中的结果集和输出参数的值的问题,问题描述如下:1,用SqlCommand.ExecuteReader(),在返回的SqlDataReader中,能取到结果集,但是取不到存储过程中的输出参数.2,用SqlCommand.ExecuteNonQuery()方法执行,可以取存储过程中的返回值,但是结果集从哪里来?先上代码如下:sqlcreateproc[GetAll(@SelectIdint,@AllCountintoutput--输入输出参数)asbeng

.net调用mysql存储过程中输出参数的问题,求救~~

问题描述 环境.net2.0+mysql5.0.37+Connector/Net5.2问题:在ASP.NET中调用mysql存储过程(输出参数名于存储过程输出参数名不同时),提示输出参数不存在:Parameter'@param1'notfoundinthecollection.当ASP.NET中修改输出参数名于存储过程中输出参数名相同的时候,提示一个新的参数未定义:Parameter'@1118079786param1'mustbedefined.而这个参数变量是自动生成的,在存储过程中我并没有

C#中调用输入输出参数的存储过程

关于存储过程的介绍大家想必也不陌生了吧.大家可以参考<浅谈存储过程>一文,这里不做介绍. 今天要介绍的是在C#里如何调用带输入输出参数的存储过程.譬如实现简单的登陆功能,根据用户输入的用户名.用户密码及用户权限实现登陆.这里充当输入参数的有:用户名.用户密码.登陆权限.充当输出参数的就是在数据库里面根据输入参数的信息查询数据库中是否有记录.具体来说就是记录的行数. 数据库中的存储过程构建代码: Code 1use Hotel --数据库名 2if exists(select * from sy

存储过程的输出参数,返回值与结果集_MsSql

每个存储过程都有默认的返回值,默认值为0.下面我们分别看看在management studio中如何查看输出参数,返回值以及结果集,然后我们再在ASP.NET调用存储过程中如何获得输出参数,返回值以及结果集. 首先:在sql server management studio中查看输出参数,返回值以及结果集.本示例以Northwind数据库为例. 复制代码 代码如下: create proc Employee @Rowcount int=0 output as begin SELECT * FRO

存储过程的输出参数,返回值与结果集

每个存储过程都有默认的返回值,默认值为0.下面我们分别看看在management studio中如何查看输出参数,返回值以及结果集,然后我们再在ASP.NET调用存储过程中如何获得输出参数,返回值以及结果集. 首先:在sql server management studio中查看输出参数,返回值以及结果集.本示例以Northwind数据库为例. 复制代码 代码如下: create proc Employee @Rowcount int=0 output as begin SELECT * FRO

JAVA中对存储过程的调用方法(二) 带输出参数的

二.带输出参数的 1:返回int -------------------------带输出参数的---------------- alter procedure getsum @n int =0, @result int output as declare @sum int declare @i int set @sum=0 set @i=0 while @i<=@n begin set @sum=@sum+@i set @i=@i+1 end set @result=@sum --------

JAVA中对存储过程的调用方法(一) 不带输出参数的

一.不带输出参数的 ---------------不带输出参数的---------------------------------- create procedure getsum @n int =0<--此处为参数--> as declare @sum int<--定义变量--> declare @i int set @sum=0 set @i=0 while @i<=@n begin set @sum=@sum+@i set @i=@i+1 end print 'the

Oracle中自定义函数 无参数 输入参数 输出参数

-- 19-1:建立无参数的函数 CREATE OR REPLACE FUNCTION cur_datetime RETURN VARCHAR2 IS BEGIN RETURN TO_CHAR(sysdate, 'YYYY"年"MM"月"DD"日"HH24″时"MI"分"SS"秒"'); END; / -- 19-2:建立带有输入参数的函数 CREATE OR REPLACE FUNCTION g

ExecuteReader在执行有输出参数的存储过程时拿不到输出参数

异常处理汇总-后端系列 http://www.cnblogs.com/dunitian/p/4523006.html 后期会在博客首发更新:http://dnt.dkill.net/Article/Detail/312 今天一同志问我这个问题,这个是过程还原: 调用SQLHelper的时候发现输出参数没值了??? 不用sqlhelper也是没有?神马情况? 用sqldataadapter却可以? 吓死宝宝了,赶紧看看啥情况.先换种方法看看 ,,,我去,可以哇!那么是不是ExecuteReader