关于System.Data.ParameterDirection四个枚举类型所起的作用(转)
相信大家都知道.net中有四个关于参数传入传出的类型 分别是:
System.Data.ParameterDirection.Input
System.Data.ParameterDirection.InputOutput
System.Data.ParameterDirection.Output
System.Data.ParameterDirection.ReturnValue
感官上理解就是只能传入 即可以传入又可以传出 只能传出 和 返回值 实际应用中和感官的理解一致吗?我也不大清楚 反正以前做的系统都没有遇见问题 所以也没有把这几个参数搞的很明白 不过心中始终有疑问 所以今天就抽了点时间做了一个例子把原理搞清楚
首先我把.Net中的参数定义为形式参数 而把存储过程的参数定义为实际参数
比如:cmd.Parameters.Add("@Input", System.Data.SqlDbType.Int); @Input为形式参数
而存储过程的@Input int, @Input为实际参数
得到的结论如下:
数据库存储过程的实际参数如果没有默认值则形式参数必须传值给实际参数
但是如果形式参数的类型为ParameterDirection.Output 则传给实际参数的永远是空值
如果形式参数的类型为ParameterDirection.ReturnValue 则形式参数不会传值给实际参数 实际参数必须有默认值 否则代码会报错
如果形式参数类型为ParameterDirection.InputOutput 或者 ParameterDirection.Output 则实际参数必须有output 关键字
另外需要注意的是在.net中 System.DBNull.Value表示数据库参数为空值 而不是null
用于测试的存储过程如下:
/*
function:测试C#中 ParameterDirection 枚举类的作用
coder:benniuniu
time:2007-11-25
*/
CREATE PROCEDURE proc_test_SQLParametersValue
@Input int,
@InputOutput int output,
@Output int output,
@ReturnValue int=1
AS
select @Input as Input,@InputOutput as InputOutput,@Output as [Output],@ReturnValue as ReturnValue
return 2
测试用的C# 代码类如下:
using System;
using System.Collections.Generic;
using System.Text;
using System.Data.SqlClient;
using System.Configuration;
using System.Data;
namespace CoderHelper.Test
{
class SQLParametersValue
{
public SQLParametersValue()
{
}
public int? Input=1;
public int? InputOutput=2;
public int? Output=3;
public int? ReturnValue=4;
public int? Input2;
public int? InputOutput2;
public int? Output2;
public int? ReturnValue2;
public void GetData(DataTable dt)
{
SqlConnection conn = newSqlConnection(ConfigurationSettings.AppSettings["LocalSqlConnectionString"]);
SqlCommand cmd = new SqlCommand();
cmd.Connection = conn;
cmd.CommandType = System.Data.CommandType.StoredProcedure;
cmd.CommandText = "proc_test_SQLParametersValue";
cmd.Parameters.Add("@Input", System.Data.SqlDbType.Int);
cmd.Parameters["@Input"].Direction = System.Data.ParameterDirection.Input;
if (Input == null)
{
cmd.Parameters["@Input"].Value = System.DBNull.Value;
}
else
{
cmd.Parameters["@Input"].Value = Input;
}
cmd.Parameters.Add("@InputOutput", System.Data.SqlDbType.Int);
cmd.Parameters["@InputOutput"].Direction = System.Data.ParameterDirection.InputOutput;
if (InputOutput == null)
{
cmd.Parameters["@InputOutput"].Value = System.DBNull.Value;
}
else
{
cmd.Parameters["@InputOutput"].Value = InputOutput;
}
cmd.Parameters.Add("@Output", System.Data.SqlDbType.Int);
cmd.Parameters["@Output"].Direction = System.Data.ParameterDirection.Output;
if (Output == null)
{
cmd.Parameters["@Output"].Value = System.DBNull.Value;
}
else
{
cmd.Parameters["@Output"].Value = Output;
}
cmd.Parameters.Add("@ReturnValue", System.Data.SqlDbType.Int);
cmd.Parameters["@ReturnValue"].Direction = System.Data.ParameterDirection.ReturnValue;
if (ReturnValue == null)
{
cmd.Parameters["@ReturnValue"].Value = System.DBNull.Value;
}
else
{
cmd.Parameters["@ReturnValue"].Value = ReturnValue;
}
SqlDataAdapter sa = new SqlDataAdapter(cmd);
try
{
if (conn.State == System.Data.ConnectionState.Closed)
{
conn.Open();
}
sa.Fill(dt);
if (cmd.Parameters["@Input"].Value != System.DBNull.Value)
Input2 = Convert.ToInt32(cmd.Parameters["@Input"].Value);
if (cmd.Parameters["@InputOutput"].Value != System.DBNull.Value)
InputOutput2 = Convert.ToInt32(cmd.Parameters["@InputOutput"].Value);
if (cmd.Parameters["@Output"].Value != System.DBNull.Value)
Output2 = Convert.ToInt32(cmd.Parameters["@Output"].Value);
if (cmd.Parameters["@ReturnValue"].Value != System.DBNull.Value)
ReturnValue2 = Convert.ToInt32(cmd.Parameters["@ReturnValue"].Value);
}
catch (Exception ex)
{
}
finally
{
conn.Close();
}
}
}
}
如上代码
public int? Output=3; 但是实际传给存储过程的值还是空值
public int? ReturnValue=4; 但是实际没有传值给存储过程
ReturnValue2 = Convert.ToInt32(cmd.Parameters["@ReturnValue"].Value); 取的是存储过程return的返回值此例存储过程代码是return 2 所以取得值是2。
注意:
1:return只能是int类型,有很大的局限性,所以Procedure返回值一般使用output。
2:Procedure中output和out是一样的,大多数网上写的是错的
3:ParameterDirection(Input,Output,InputOutput,ReturnValue)中的Output,InputOutput只是代表C#怎么传值给Procedure,Output传空,InputOutput传实际的值
关于System.Data.ParameterDirection四个枚举类型所起的作用(转)的更多相关文章
- 使用EF访问数据库,出现“System.Data.Entity.Internal.AppConfig”的类型初始值设定项引发异常。
今天在使用的EF时候,发生了"System.Data.Entity.Internal.AppConfig"的类型初始值设定项引发异常.这样的一个错误 查了原因,原来是appconf ...
- “System.Data.Entity.Internal.AppConfig"的类型初始值设定项引发异常。{转}
<connectionStrings> <add name="ConnectionStringName" providerName="System.Da ...
- System.Data.Entity.Internal.AppConfig"的类型初始值设定项引发异常
在学习EF code First的小案例的时候,遇见了这个异常 <configSections> <!-- For more information on Entity Framew ...
- Entityframework:“System.Data.Entity.Internal.AppConfig”的类型初始值设定项引发异常。
<configSections> <!-- For more information on Entity Framework configuration, visit http:// ...
- Language Guide (proto3) | proto3 语言指南(四)枚举类型
枚举类型 定义消息类型时,可能希望其中一个字段只包含预定义值列表中的一个.例如,假设您想为每个SearchRequest添加一个corpus(语料库)字段,其中语料库的值可以是UNIVERSAL.WE ...
- EF中System.Data.Entity.Internal.AppConfig的类型初始值设定项引发异常
使用Entity的时候遇到的一个错 问题出在项目的App.config中 解决: 1.configSections要写在最顶端 2. 当中的incariantName会变成incariantNodeN ...
- System.Data.Entity.Internal.AppConfig 类型初始值设定项引发异常
在一开始时将connectionStrings 写在了configSections之上如下图一示,结果抛出:“System.Data.Entity.Internal.AppConfig”的类型初始值设 ...
- Method 'ExecuteAsync' in type 'System.Data.Entity.SqlServer.DefaultSqlExecutionStrategy' does not have an implementation
一.错误信息 Entity Framework 6.0数据迁移:Add-Migration XXXX 命令发生错误 System.Reflection.TargetInvocationExceptio ...
- C#枚举类型的常用操作总结
枚举类型是定义了一组"符号名称/值"配对.枚举类型是强类型的.每个枚举类型都是从system.Enum派生,又从system.ValueType派生,而system.ValueTy ...
随机推荐
- NWU现场赛——解题报告
负二进制转换 Time Limit : 3000/1000ms (Java/Other) Memory Limit : 65535/32768K (Java/Other) Problem Desc ...
- Android学习--持久化(一) 文件存储
持久化之 文件存储 这里把Android持久化全都整理一下,这一篇文章先简单的说一下文件的存储,通过下面一个简单的Demo,理解一下这个文件存储,先说说下面Demo的思路: 1.创建EditTex ...
- Delphi中@,^,#,$分别表示什么?
@:取址运算符; var int:integer; p:^integer; new(P); int:=24; p:=@int; dispose(P); ^:指针的引用解析操作符; var pint:^ ...
- ios开发小结之app发布升级
在近两个月的开发中,遇到了挺多问题的,几天加班加点,最后还是在年前发布并更新了一个版本,欢迎下载无觅下载. 最头疼的问题是提交app审核,之前的工程不太规范,导致一些文件icon没有设置好,直接val ...
- devicemapper: Error running deviceCreate (ActivateDevice) dm_task_run failed
在一台新机子上面,docker处理完lvs数据卷之后,启动docker服务时,出现了启动失败,失败信息如下: [root@hxin221 ~]# systemctl status docker ● d ...
- Android BroadcastReceiver 注册和反注册
说起来这个问题很简单,只要注册和反注册成对出现就行,好像很多教材都是如此介绍.但实际开发中,对广播注册和反注册的时机把握还是很重要的. 关于广BroadcastReceiver注册和反注册时机,主要有 ...
- mac 安装 word2016并破解
http://blog.csdn.net/zqbx7/article/details/53448280
- hduoj1285确定比赛名次
确定比赛名次 Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others) Total ...
- java基础篇5之泛型
1 泛型的基本应用 //反射方式 指定类型,就不用强转 Construcctor<String> constructor = String.class.getConstructor(Str ...
- js 快速排序
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...