Winform调用存储过程
数据表及数据准备:
create table Member
(
MemberId int primary key identity(1,1),
MemberAccount nvarchar(20) unique,
MemberPwd nvarchar(20),
MemberName nvarchar(20),
MemberPhone nvarchar(20)
)
truncate table Member
insert into Member(MemberAccount,MemberPwd,MemberName,MemberPhone)
values('liubei','123456','刘备','4659874564')
insert into Member(MemberAccount,MemberPwd,MemberName,MemberPhone)
values('guanyu','123456','关羽','42354234124')
insert into Member(MemberAccount,MemberPwd,MemberName,MemberPhone)
values('zhangfei','123456','张飞','41253445')
insert into Member(MemberAccount,MemberPwd,MemberName,MemberPhone)
values('zhangyun','123456','赵云','75675676547')
insert into Member(MemberAccount,MemberPwd,MemberName,MemberPhone)
values('machao','123456','马超','532523523')
本文以下内容都借助于前面封装的DBHelper类
一、调用exec语句执行存储过程
由于在SQL SERVER内部调用存储过程使用的方式是:
exec 存储过程名 参数1,参数2,参数3...
所以我们可以在C#中调用exec的sql语句,让此sql语句去调用存储过程,严格来说,此种方式并不能称之为C#调用存储过程,本质上仍然是调用的sql语句。
示例:


需求:采用调用存储过程的方式实现数据的显示以及数据的新增。
主要代码:
SQL存储过程代码:
--查询Member表所有数据的存储(没有参数)
create proc procSelectMember
as
select * from Member
go
exec procSelectMember
--添加会员信息(有输入参数)
create proc procInsertMember
@acc nvarchar(20),
@pwd nvarchar(20),
@memName nvarchar(20),
@memPhone nvarchar(20)
as
insert into Member(MemberAccount,MemberPwd,MemberName,MemberPhone)
values(@acc,@pwd,@memName,@memPhone)
go
exec procInsertMember 'sunwukong','123456','孙悟空','13554856985'
数据显示C#代码:
private void BindData()
{
DBHelper.PrepareSql("exec procSelectMember");
this.dataGridView1.DataSource = DBHelper.ExecQuery();
}
private void Form1_Load(object sender, EventArgs e)
{
BindData();
}
数据新增C#代码:
private void btAdd_Click(object sender, EventArgs e)
{
DBHelper.PrepareSql(string.Format("exec procInsertMember '{0}','{1}','{2}','{3}'"
,this.txtAccount.Text,this.txtPwd.Text,this.txtNickName.Text,this.txtPhone.Text));
DBHelper.ExecNonQuery();
}
二、直接调用存储过程
调用存储过程需要将CommandType执行命令类型设置为CommandType.StoredProcedure存储过程。
(1)调用没有参数的存储过程

需求:实现数据的显示。
主要代码:
SQL存储过程代码:
--查询Member表所有数据的存储(没有参数)
create proc procSelectMember
as
select * from Member
go
--调用
exec procSelectMember
为了支持存储过程,给DBHelper添加方法:
public static void PrepareProc(string sql)
{
OpenConn(); //打开数据库连接
adp = new SqlDataAdapter(sql, conn);
adp.SelectCommand.CommandType = CommandType.StoredProcedure;
}
窗体代码:
private void BindData()
{
DBHelper.PrepareProc("procSelectMember");
this.dataGridView1.DataSource = DBHelper.ExecQuery();
}
private void Form1_Load(object sender, EventArgs e)
{
BindData();
}
(2)调用有输入参数的存储过程

需求:实现数据的新增。
主要代码:
SQL存储过程代码:
--添加会员信息(有输入参数)
create proc procInsertMember
@acc nvarchar(20),
@pwd nvarchar(20),
@memName nvarchar(20),
@memPhone nvarchar(20)
as
insert into Member(MemberAccount,MemberPwd,MemberName,MemberPhone)
values(@acc,@pwd,@memName,@memPhone)
go
--调用
exec procInsertMember 'sunwukong','123456','孙悟空','13554856985'
窗体代码:
private void btAdd_Click(object sender, EventArgs e)
{
DBHelper.PrepareProc("procInsertMember");
DBHelper.SetParameter("acc", this.txtAccount.Text);
DBHelper.SetParameter("pwd",this.txtPwd.Text);
DBHelper.SetParameter("memName", this.txtNickName.Text);
DBHelper.SetParameter("memPhone", this.txtPhone.Text);
DBHelper.ExecNonQuery();
}
(3)调用有输入和输出参数的存储过程

需求:输入用户名,点击"查询电话"按钮,在下面显示姓名和号码。
主要代码:
SQL存储过程:
--根据账号查询姓名和电话(有输入参数,有输出参数)
create proc procGetInfoByAcc
@acc nvarchar(20),
@memName nvarchar(20) output,
@phone nvarchar(20) output
as
select @memName = (select MemberName from Member where MemberAccount=@acc)
select @phone = (select MemberPhone from Member where MemberAccount=@acc)
go
--调用
declare @name nvarchar(20)
declare @phone nvarchar(20)
exec procGetInfoByAcc 'machao',@name output,@phone output
select @name,@phone
为了支持输出参数,给DBHelper添加方法:
/// <summary>
/// 设置输出参数(不指定长度,适合非字符串)
/// </summary>
/// <param name="parameterName">参数名称</param>
/// <param name="dbType">参数类型</param>
public static void SetOutParameter(string parameterName, SqlDbType dbType)
{
parameterName = "@" + parameterName.Trim();
SqlParameter parameter = new SqlParameter(parameterName, dbType);
parameter.Direction = ParameterDirection.Output;
adp.SelectCommand.Parameters.Add(parameter);
}
/// <summary>
/// 设置输出参数(指定长度,适合字符串)
/// </summary>
/// <param name="parameterName">参数名称</param>
/// <param name="dbType">参数类型</param>
/// <param name="size">参数长度</param>
public static void SetOutParameter(string parameterName, SqlDbType dbType, int size)
{
parameterName = "@" + parameterName.Trim();
SqlParameter parameter = new SqlParameter(parameterName, dbType, size);
parameter.Direction = ParameterDirection.Output;
adp.SelectCommand.Parameters.Add(parameter);
}
/// <summary>
/// 获取参数内容值
/// </summary>
/// <param name="parameterName">参数名称</param>
/// <returns>参数值</returns>
public static object GetParameter(string parameterName)
{
parameterName = "@" + parameterName.Trim();
return adp.SelectCommand.Parameters[parameterName].Value;
}
窗体代码:
private void btSearch_Click(object sender, EventArgs e)
{
DBHelper.PrepareProc("procGetInfoByAcc");
DBHelper.SetParameter("acc", this.txtAccount.Text);
DBHelper.SetOutParameter("memName", SqlDbType.NVarChar, 20);
DBHelper.SetOutParameter("phone", SqlDbType.NVarChar, 20);
DBHelper.ExecNonQuery();
this.lblName.Text = "姓名:" + DBHelper.GetParameter("memName").ToString();
this.lblPhone.Text = "电话:" + DBHelper.GetParameter("phone").ToString();
}
(4)调用有输入输出参数的存储过程

需求:密码升级,传入用户名和密码;如果用户名密码正确,并且密码长度<8,自动升级成8位密码。
主要代码:
SQL存储过程(SQLSERVER中output参数直接传入值即可以做输入参数,也可以做输出参数):
--密码升级,传入用户名和密码,如果用户名密码正确,并且密码长度<8,自动升级成8位密码
--有输入输出参数(密码作为输入参数也作为输出参数)
select FLOOR(RAND()*10) --0-9之间随机数
create proc procPwdUpgrade
@acc nvarchar(20),
@pwd nvarchar(20) output
as
if not exists(select * from Member where MemberAccount=@acc and MemberPwd=@pwd)
set @pwd = ''
else
begin
if len(@pwd) < 8
begin
declare @len int = 8- len(@pwd)
declare @i int = 1
while @i <= @len
begin
set @pwd = @pwd + cast(FLOOR(RAND()*10) as varchar(1))
set @i = @i+1
end
update Member set MemberPwd = @pwd where MemberAccount=@acc
end
end
go
--调用
declare @pwd nvarchar(20) = '123456'
exec procPwdUpgrade 'liubei',@pwd output
select @pwd
为了支持输入输出参数,给DBHelper添加方法:
/// <summary>
/// 设置输入输出参数(不指定长度,适合非字符串)
/// </summary>
/// <param name="parameterName">参数名称</param>
/// <param name="dbType">参数类型</param>
public static void SetInOutParameter(string parameterName, SqlDbType dbType, object parameterValue)
{
parameterName = "@" + parameterName.Trim();
SqlParameter parameter = new SqlParameter(parameterName, dbType);
parameter.Value = parameterValue;
parameter.Direction = ParameterDirection.InputOutput;
adp.SelectCommand.Parameters.Add(parameter);
}
/// <summary>
/// 设置输入输出参数(指定长度,适合字符串)
/// </summary>
/// <param name="parameterName">参数名称</param>
/// <param name="dbType">参数类型</param>
/// <param name="size">参数长度</param>
public static void SetInOutParameter(string parameterName, SqlDbType dbType, int size, object parameterValue)
{
parameterName = "@" + parameterName.Trim();
SqlParameter parameter = new SqlParameter(parameterName, dbType, size);
parameter.Value = parameterValue;
parameter.Direction = ParameterDirection.InputOutput;
adp.SelectCommand.Parameters.Add(parameter);
}
窗体代码:
//密码升级,传入用户名和密码,
//如果用户名密码正确,并且密码长度<8,自动升级成8位密码
private void btUpgrade_Click(object sender, EventArgs e)
{
DBHelper.PrepareProc("procPwdUpgrade");
DBHelper.SetParameter("acc", this.txtAccount.Text);
DBHelper.SetInOutParameter("pwd", SqlDbType.NVarChar, 20, this.txtPwd.Text);
DBHelper.ExecNonQuery();
this.lblNewPwd.Text = DBHelper.GetParameter("pwd").ToString();
}
(5)调用有返回值的存储过程
SQLSERVER存储过程返回值只能是整数。

需求:实现数据的新增,由SQLSERVER返回执行的状态。
主要代码:
SQL存储过程代码:
--添加会员信息(有返回值)
create proc procInsertMember
@acc nvarchar(20),
@pwd nvarchar(20),
@memName nvarchar(20),
@memPhone nvarchar(20)
as
insert into Member(MemberAccount,MemberPwd,MemberName,MemberPhone)
values(@acc,@pwd,@memName,@memPhone)
declare @myErr int = @@error
if @myErr = 0
return 1
else if @myErr = 2627 --唯一约束
return -1
else
return -100
go
--调用
declare @return int
exec @return = procInsertMember 'sunwukong','123456','孙悟空','13554854785'
print @return
为了支持返回值,给DBHelper添加方法:
/// <summary>
/// 设置返回值参数
/// </summary>
/// <param name="parameterName">参数名称</param>
public static void SetReturnParameter(string parameterName)
{
parameterName = "@" + parameterName.Trim();
SqlParameter parameter = new SqlParameter();
parameter.ParameterName = parameterName;
parameter.Direction = ParameterDirection.ReturnValue;
adp.SelectCommand.Parameters.Add(parameter);
}
窗体代码:
private void btAdd_Click(object sender, EventArgs e)
{
try
{
DBHelper.PrepareProc("procInsertMember");
DBHelper.SetParameter("acc", this.txtAccount.Text);
DBHelper.SetParameter("pwd", this.txtPwd.Text);
DBHelper.SetParameter("memName", this.txtNickName.Text);
DBHelper.SetParameter("memPhone", this.txtPhone.Text);
DBHelper.SetReturnParameter("returnValue");
DBHelper.ExecNonQuery();
int result = (int)DBHelper.GetParameter("returnValue");
if (result == 1)
MessageBox.Show("添加成功!");
}
catch (Exception ex)
{
int result = (int)DBHelper.GetParameter("returnValue");
if (result == -1)
MessageBox.Show("用户名重名了,违反了唯一约束!");
if (result == -100)
MessageBox.Show(ex.Message);
}
}
Winform调用存储过程的更多相关文章
- spring data jpa 调用存储过程
网上这方面的例子不是很多,研究了一下,列出几个调用的方法. 假如我们有一个mysql的存储过程 CREATE DEFINER=`root`@`localhost` PROCEDURE `plus1in ...
- myabatis oracle 调用存储过程返回list结果集
Mapper.xml 配置 <resultMap type="emp" id="empMap"> <id property="emp ...
- C#winForm调用WebService的远程接口
Web Service 的创建简单编码.发布和部署 上一篇详细概述了WebService的创建,编码,发布和部署,那么作为客户端的程序如何访问远程端的WebService 接下来看一下具体步骤: ...
- IBatis.Net使用总结(四)-- IBatis 调用存储过程
IBatis 调用存储过程 http://www.cnblogs.com/jeffwongishandsome/archive/2010/01/10/1543219.html http://www.c ...
- SQL SERVER使用ODBC 驱动建立的链接服务器调用存储过程时参数不能为NULL值
我们知道SQL SERVER建立链接服务器(Linked Server)可以选择的驱动程序非常多,最近发现使用ODBC 的 Microsoft OLE DB 驱动程序建立的链接服务器(Linked S ...
- 【Java EE 学习 29 下】【JDBC编程中操作Oracle数据库】【调用存储过程的方法】
疑问:怎样判断存储过程执行之后返回值是否为空. 一.连接oracle数据库 1.需要的jar包:在安装的oracle中就有,所以不需要到官网下载,我的oracle11g下:D:\app\kdyzm\p ...
- MyBatis学习总结(六)——调用存储过程(转载)
本文转载自:http://www.cnblogs.com/jpf-java/p/6013518.html 一.提出需求 查询得到男性或女性的数量, 如果传入的是0就女性否则是男性 二.准备数据库表和存 ...
- C# 调用存储过程操作 OUTPUT参数和Return返回值
本文转载:http://www.cnblogs.com/libingql/archive/2010/05/02/1726104.html 存储过程是存放在数据库服务器上的预先编译好的sql语句.使用存 ...
- jdbc调用存储过程和函数
1.调用存储过程 public class CallOracleProc { public static void main(String[] args) throws Exception{ Stri ...
随机推荐
- 「NOI十联测」深邃
「NOI十联测」深邃 要使得最大的连通块最小,显然先二分答案. 先固定1结点为根. 对于一个果实,显然是先处理子树中未分配的点,再向外延伸. 每个结点记录一个\(si[]\),表示子树中未分配的点数, ...
- response.getWriter().write()和 response.getWriter().print()的区别 以及 PrintWriter对象 和 out对象 的区别
感谢原文作者:krismile__qh 原文链接:https://blog.csdn.net/krismile__qh/article/details/89926001 一.response.getW ...
- 创建SSH密钥时使用了自定义文件名遇到的问题
问题描述 如图,我自定义了密钥文件名字. 所以在测试连接时导致了: 问题解决 连接的时候指定自己重命名的私钥文件名就好了. ssh -T -i git_test git@github.com SSH命 ...
- 对战平台虚拟War3局域网的原理对战平台虚拟War3局域网的原理
转载请注明来源:https://www.cnblogs.com/hookjc/ 以War3为例,启动魔兽后,首先是如何看见主机的问题:魔兽是通过TCP/UDP协议进行数据发送的,那如何实现看到对方?我 ...
- 冒泡法排序_c++实现
看完了郝斌老师的c语言视频,冒泡法排序,就试着写了.我觉得学习算法最重要的不是代码,而是它的原理. 代码: /** 2 * Copyright (c) 1991 - 2016 Arvin Tang. ...
- TestNG--@Factory
原文地址:http://blog.csdn.net/wanghantong TestNg的@Factory注解从字面意思上来讲就是采用工厂的方法来创建测试数据并配合完成测试 其主要应对的场景是:对于某 ...
- Keycloak 团队宣布他们正在弃用大多数 Keycloak 适配器,包括Spring Security和Spring Boot
2月14日,Keycloak 团队宣布他们正在弃用大多数 Keycloak 适配器. 其中包括Spring Security和Spring Boot的适配器,这意味着今后Keycloak团队将不再提供 ...
- 手写RPC框架(六)整合Netty
手写RPC框架(六)整合Netty Netty简介: Netty是一个基于NIO的,提供异步,事件驱动的网络应用工具,具有高性能高可靠性等特点. 使用传统的Socket来进行网络通信,服务端每一个连接 ...
- CentOS7编译安装升级openssh8.7p1
因生成环境服务器安全扫描出的漏洞问题,只能升级最新的openssh,适用于centos6和centos7的升级使用. 一.编译前工作 openssl版本要求1.0.1以上,zlib版本要求1.1.4以 ...
- Java 线程的 5 种状态
线程状态图: 线程共包括以下 5 种状态: 1. 新建状态(New): 线程对象被创建后,就进入了新建状态.例如,Thread thread = new Thread(). 2. 就绪状态(Runna ...