下面的方法是我在实际开发中摸索出来的,可以在很大程度上简化调用存储过程的代码。
首先来看一下C#调用存储过程的一般过程:
1、打开数据库连接SqlConnection;
2、生成一个SqlCommand;
3、向命令对象填充参数;
4、执行存储过程;
5、关闭连接;
6、其他操作。
我这里讲的主要是简化第3步操作,最终在调用存储过程的时候只需要传递存储过程的名字和相应的参数值。调用示例如下:
dbAccess.run("p_am_deleteFile", new object[]{LoginId, Request.UserHostAddress, fileId});
由于在填充参数的时候必须要两个值,一个是参数的名字,一个是参数的值。参数值是由外部传入的,不用考虑;而参数名称是和存储过程相关的东西,应该可以由存储过程名称来确定而不用每次调用的时候写上一遍。对于这个问题,如果能将存储过程的参数保存到一个全局的地方,那么在调用存储过程的时候只要能根据存储过程的名字去索引就可以了。具体实现的时候我是将这些信息保存在数据库访问组件里面,采用名字/值对的方式。代码如下:
public class InfoTable : NameObjectCollectionBase
{
public object this[string key]
{
get
{
return(this.BaseGet(key));
}
set
{
this.BaseSet(key, value);
}
}
}
。。。。。。
protected static InfoTable procInfoTable = new InfoTable();
。。。。。。
public static InfoTable ProcInfoTable
{
get
{
return procInfoTable;
}
}
这样的话,在实际调用存储过程的时候就只需要去查这张表就可以知道存储过程的参数名了。实现代码如下:
public DataTable run(string procName, object[] parms, ref int retValue)
{
string[] paramInfo = (string[])(procInfoTable[procName]);
if (paramInfo == null)
{
ErrorInfo.setErrorInfo("未取得" + procName + "的参数!");
return null;
}
bool bOpened = (dbConn.State == ConnectionState.Open);
if (!bOpened && !connect())
{
return null;
}
DataSet ds = new DataSet();
try
{
SqlCommand cmd = new SqlCommand(procName, dbConn);
cmd.CommandType = CommandType.StoredProcedure;
for (int i = 0; i < parms.Length && i < paramInfo.Length; ++i)
{
cmd.Parameters.Add(new SqlParameter(paramInfo[i], parms[i]));
}
SqlParameter parmsr = new SqlParameter("return", SqlDbType.Int);
parmsr.Direction = ParameterDirection.ReturnValue;
cmd.Parameters.Add(parmsr);
SqlDataAdapter adp = new SqlDataAdapter(cmd);
adp.Fill(ds);
retValue = (int)(cmd.Parameters["return"].Value);
}
catch (Exception ex)
{
ErrorInfo.setErrorInfo(ex.Message);
retValue = -1;
}
if (!bOpened)
close();
if (ds.Tables.Count > 0)
return ds.Tables[0];
else
return null;
}
可以看出,每个存储过程的参数列表存储为了一个string[]。接下来的工作就是将系统里头许许多多的存储过程的参数填充到表ProcInfoTable中。我所用的数据库是Sql Server 2000,下面给出一个存储过程来解决这个烦人的问题:
create PROCEDURE dbo.p_am_procInfo
(
@procNamet_str64--存储过程的名字
)
AS
begin
set nocount on
if @procName = '' begin
select name as procName
from sysobjects
where substring(sysobjects.name, 1, 5) = 'p_am_'
end
else begin
select
syscolumns.name as paramName
from sysobjects, syscolumns
where sysobjects.id = syscolumns.id
and sysobjects.name = @procName
order by colid
end
end
这个存储过程有两个作用,在没有传递存储过程的名字的时候,该存储过程返回所有以”p_am_”开头的存储过程的名字;在传入了相应的存储过程名字后,该存储过程返回该存储过程的参数列表。这样一来,我们在程序开始的地方就可以将系统里的存储过程参数列表取出来并保存到数据库访问组件的ProcInfoTable属性中了。具体代码如下:
span.DBAccess dbAccess = new span.DBAccess();
//
//构造取存储过程的参数表
//
span.DBAccess.ProcInfoTable["p_am_procInfo"] = new string[]{"@procName"};
//
//取得其他存储过程列表
//
DataTable dt = dbAccess.run("p_am_procInfo", new object[]{""});
if (dt == null || dt.Rows.Count <= 0)
{
return;
}
//
//取得其他存储过程的参数表
//
foreach (DataRow dr in dt.Rows)
{
DataTable dtParams = dbAccess.run("p_am_procInfo", new object[]{dr["procName"]});
if (dtParams != null)
{
string[] paramInfo = new string[dtParams.Rows.Count];
for (int i = 0; i < dtParams.Rows.Count; ++i)
paramInfo[i] = dtParams.Rows[i]["paramName"].ToString();
span.DBAccess.ProcInfoTable[dr["procName"].ToString()] = paramInfo;
}
}
至此,全部技术细节介绍完毕。另外,数据库访问对象的几个接口函数也一并给出:
//打开、关闭数据库连接
public bool connect(string strConn)
public bool connect()
public bool close()
//执行SQL命令(只有一个int返回)
public int exec(string procName, object[] parms)
public int exec(string sql)
//运行(返回一个DataTable)
public DataTable run(string procName, object[] parms, ref int retValue)
public DataTable run(string procName, object[] parms)
public DataTable run(string sql)
//分页查询(页号从1开始,返回一个DataTable)
public DataTable pageQuery
(
string selectCmd,
int pageSize,
int pageNumber
)
分享到:
相关推荐
C#执行存储过程的简化-.NET教程,C#语言.docx
内容概要:简单的C# winform调用存储过程实例,创建存储过程入参,通过SqlConnection对象和SqlCommand对象调用存储过程,获取存储过程的出参并显示出来,详细代码注释,希望对用到C#调用存储过程的小伙伴有帮助 ...
它还极大地简化了使用过滤器执行 CRUD 操作、执行完整查询和执行存储过程。 它支持框架、标准和核心中的异步和非异步。快速浏览: (向下滚动查看完整的工作示例。) Dapper.SimpleRepository可以严格使用或弱类型...
17.2 文件存储管理 .217 17.3 读 写 文 件 .222 17.4 异步文件操作 .227 17.5 小 结 .234 第十八章 高 级 话 题 .235 18.1 注册表编程 .235 18.2 在 C #代码中调用 C++和 VB 编写的组件 .240 18.3 版 ...
C#5.0本质论第四版,高清扫描的,对C#5.0技术讲的比较详细,第1章 C#概述 1 1.1 Hello,World 1 1.2 C#语法基础 3 1.2.1 C#关键字 3 1.2.2 标识符 4 1.2.3 类型定义 5 1.2.4 Main 6 1.2.5...
C#编译原理 目 录 译者序 前言 第1章 概论 1 1.1 为什么要用编译器 2 1.2 与编译器相关的程序 3 1.3 翻译步骤 5 1.4 编译器中的主要数据结构 8 1.5 编译器结构中的其他问题 10 1.6 自举与移植 12 1.7 TINY样本语言与...
迭代子模式简化了聚集 的界面。每一个聚集对象都可以有一个或一个以上的迭代子对象,每一 个迭代子的迭代状 态可以是彼此独立的。迭代算法可以独立于聚集角色变化。 17、MEDIATOR —四个 MM 打麻将,相互之间谁...
5.8 通过CountdownEvent简化动态fork和join场景 5.9 使用原子操作 5.10 小结 第6章 PLINQ:声明式数据并行 6.1 从LINQ转换到PLINQ 6.1.1 ParallelEnumerable及其AsParallel方法 6.1.2 AsOrdered和orderby子句 ...
5.8 通过CountdownEvent简化动态fork和join场景 5.9 使用原子操作 5.10 小结 第6章 PLINQ:声明式数据并行 6.1 从LINQ转换到PLINQ 6.1.1 ParallelEnumerable及其AsParallel方法 6.1.2 AsOrdered和orderby子句 ...
传统的asp里这并不是一个简单的任务,但在asp.net中,DataGrid控件把这一过程简化为只有 几行代码。因此,在 asp.net中,分页很简单,但是默认的DataGrid分页事件会从数据库中把所 有的记录全部读出来放到asp.net ...
无法执行自定义SQL 简化的ORM,仅从DbDataReader映射对象属性能够以不同于SQL返回的顺序指定代码中的字段。 在DbContext内部自动生成DbSet,因为在使用存储过程时,这很可能是负担。 未实现FormattableString支持。...
以SQL Server为工具,讲解SQL语言的应用,提供了近500个曲型应用,读者可以随查随用,深入讲解SQL语言的各种查询语句,详细介绍数据库设计及管理,详细讲解存储过程、解发器和游标等知识,讲解了SQL语言在高级语言中...