SQL 存储过程入门(事务)(四)
SQL 存储过程入门(事务)(四)
本篇我们来讲一下事务处理技术。
为什么要使用事务呢,事务有什么用呢,举个例子。
假设我们现在有个业务,当做成功某件事情的时候要向2张表中插入数据,A表,B表,我们插入的顺序是先插入A,再插入B表,如果都顺利插入成功了,当然没有问题,如果任意一张表插入失败了,而另一张表插入成功了,插入成功的表就是垃圾数据了。我们要判断,任意一张表插入失败都回滚,就是都不插入,这就是事务的基本使用。
一,sql事务定义
所谓事务是用户定义的一个数据库操作序列,是一个不可分割的工作单位。他包含的所有数据库操作命令作为一个整体一起向系提交或撤消 ,这些操作要么全做,要么全不做,
例如在关系数据库中,一个事务可以是一条sql语句,或者是一组sql语句或者是整个程序。
二,sql事务语句
开始事务:BEGIN TRANSACTION
提交事务:COMMIT TRANSACTION
回滚事务:ROLLBACK TRANSACTION
事务通常是以begin transaction开始的,以commit或rollback 结束,commit表示提交,既提交事务的所有操作。具体的说就是将事务中所有对数据库的更新写到磁盘
上的物理数据中去,事务正常结束。
Rollback表示回滚,在事务运行的过程中发生了某种故障,事务不能继续执行,系统将事务中对数据库的所有已完成的操作全部取消,回滚到事务开始时的状态。这里的操作
指对数据库的更新操作。
三,事务的特性(ACID Propertites)
原子性(Atomicity):事务是数据库的逻辑工作单位,事务中包括的诸操作要么都做,要么都不做。
一致性(Consistency):事物完成时,数据必须是一致的,也就是说,和事物开始之前,数据存储中的数据处于一致状态。保证数据的无损
隔离性(Isolation):一个事务的执行不能被其他事务干扰。并发执行的各个事务之间不能互相干扰。
持续性(Durability):指一个事务一旦提交,它对数据库中数据的改变时永久性的。
四,存储过程使用事务
这里做好准备工作,建立一张表,插入一条数据。

- --新建表
- create table userinfo
- (ID int identity(1,1) ,
- UserName varchar(20) primary key,
- UserPwd varchar(20) ,
- RegisterTime datetime
- )
- --初始化插入一条记录
- insert into userinfo(username,userpwd,RegisterTime)
- values('admin','admin',getdate())
- select * from userinfo
- -------------------
- ID userName UserPwd RegisterTime
- 1 admin admin 2013-04-13 10:30:36.387

从表结构看出,UserName是主键,是唯一值,现在要插入两条数据

- Create Procedure MyProcedure
- AS
- Begin
- Set NOCOUNT ON;
- Set XACT_ABORT on; --这句话非常重要
- Begin Tran --开始事务
- insert into userinfo(username,userpwd,RegisterTime) values('admin','admin',getdate())
- insert into userinfo(username,userpwd,RegisterTime) values('jack','jack',getdate())
- Commit Tran --提交事务
- End

执行

- exec MyProcedure
- /*
- 消息 2627,级别 14,状态 1,过程 MyProcedure,第 9 行
- 违反了 PRIMARY KEY 约束 'PK__userinfo__C9F284577F60ED59'。不能在对象 'dbo.userinfo' 中插入重复键。
- */
- --查看数据库
- ------------------------------------------------
- 12 admin admin 2013-04-13 10:41:22.457

- 上面说了 Set XACT_ABORT on; 这句话非常重要 ,为什么呢?我们来设置为off的时候来看效果

- Create Procedure MyProcedure
- AS
- Begin
- Set NOCOUNT ON;
- Set XACT_ABORT off; --这句话非常重要
- Begin Tran --开始事务
- insert into userinfo(username,userpwd,RegisterTime) values('admin','admin',getdate())
- insert into userinfo(username,userpwd,RegisterTime) values('jack','jack',getdate())
- Commit Tran --提交事务
- End

执行并查看结果

- exec MyProcedure
- /*
- 消息 2627,级别 14,状态 1,过程 MyProcedure,第 9 行
- 违反了 PRIMARY KEY 约束 'PK__userinfo__C9F284577F60ED59'。不能在对象 'dbo.userinfo' 中插入重复键。
- 语句已终止。
- */
- --查看结果
- select * from userinfo
-------------------------------------------- 12 admin admin 2013-04-13 10:41:22.457
- 15 jack jack 2013-04-13 10:44:05.203

这里我们将XACT_ABORT 设置为off,事务中执行已经出现错误了,但是还是将 “jack”这条记录插入进去了。这就违反了事务的一致性原则了。所以我们要将XACT_ABORT 设置为ON的原因。
看看下面说明:
1 、使用存储过程执行事物,需要开启XACT_ABORT参数(默认值为Off),将该参数设置为On,表示当执行事务时,如果出错,会将transcation设置为uncommittable状态,那么在语句块批处理结束后将回滚所有操作;如果该参数设置为Off,表示当执行事务时,如果出错,出错的语句将不会执行,其他正确的操作继续执行。
2、当SET NOCOUNT 为 ON 时,不返回计数(计数表示受 Transact-SQL 语句影响的行数,例如在Sql server查询分析器中执行一个delete操作后,下方窗口会提示(3)Rows Affected)。当 SET NOCOUNT 为 OFF 时,返回计数,我们应该在存储过程的头部加上SET NOCOUNT ON 这样的话,在退出存储过程的时候加上 SET NOCOUNT OFF这样的话,以达到优化存储过程的目的。
- 五,存储过程中事务和try…catch联合使用
- 如果我们在存储过程事务中出现了错误,我们不想显示错误,我们想动态处理这些错误信息,比如出错了,我们回滚,我们设置某个属性的值,这里就会用到try ,catch了
还是从例子出发

- Create Procedure MyProcedure
- AS
- Begin
- Set NOCOUNT ON;
- Set XACT_ABORT ON; --这句话非常重要
- begin try
- Begin Tran --开始事务
- insert into userinfo(username,userpwd,RegisterTime) values('admin','admin',getdate())
- insert into userinfo(username,userpwd,RegisterTime) values('jack','jack',getdate())
- Commit Tran --提交事务
- end try
- begin catch
- --在此可以使用xact_state()来判断是否有不可提交的事务,不可提交的事务
- --表示在事务内部发生错误了。Xact_state()有三种值:-1.事务不可提交;
- --1.事务可提交;0.表示没有事务,此时commit或者rollback会报错。
- if xact_state()=-1
- rollback tran;
- end catch
- Set XACT_ABORT OFF;
- End

当我们执行的时候不会再出现刚才的那种错误了,

- exec MyProcedure
- --------------
- 命令已成功完成。 --没有出现那种错误
- select * from userinfo
- --------------------------------------------------
- 22 admin admin 2013-04-13 10:55:50.653

可以看到,事务回滚了,没有插入数据了。
- 如果我们想看到错误信息呢,再来看个例子

- Create Procedure MyProcedure
- AS
- Begin
- Set NOCOUNT ON;
- Set XACT_ABORT ON; --这句话非常重要
- begin try
- Begin Tran --开始事务
- insert into userinfo(username,userpwd,RegisterTime) values('admin','admin',getdate())
- insert into userinfo(username,userpwd,RegisterTime) values('jack','jack',getdate())
- Commit Tran --提交事务
- end try
- begin catch
- --在此可以使用xact_state()来判断是否有不可提交的事务,不可提交的事务
- --表示在事务内部发生错误了。Xact_state()有三种值:-1.事务不可提交;
- --1.事务可提交;0.表示没有事务,此时commit或者rollback会报错。
- if xact_state()=-1
- begin
- rollback tran;
- SELECT ERROR_NUMBER() AS ErrorNumber,
- ERROR_MESSAGE() AS ErrorMessage;
- end
- end catch
- End

执行
- exec MyProcedure
- --------------------------------
- ErrorNumber ErrorMessage
- 2627 违反了 PRIMARY KEY 约束 'PK__userinfo__C9F284577F60ED59'。不能在对象 'dbo.userinfo' 中插入重复键。
说明:1、捕获错误的函数有很多,如下:
ERROR_NUMBER() 返回错误号。
ERROR_SEVERITY() 返回严重性。
ERROR_STATE() 返回错误状态号。
ERROR_PROCEDURE() 返回出现错误的存储过程或触发器的名称。
ERROR_LINE() 返回导致错误的例程中的行号。
ERROR_MESSAGE() 返回错误消息的完整文本。该文本可包括任何可替换参数所提供的值,如长度、对象名或时间。
在存储过程中使用事务时,如果存在try…catch语句块,那么当捕获到错误时,需要在catch语句块中手动进行Rollback操作,否则系统会给客户端传递一条错误信息。如果在存储过程开始处将set xact_abort on,那么当有错误发生时,系统会将当前事务置为不可提交状态,即会将xact_state()置为-1,此时只可以对事务进行Rollback操作,不可进行提交(commit)操作,那么我们在catch语句块中就可以根据xact_state()的值来判断是否有事务处于不可提交状态,如果有则可以进行rollback操作了。
如果在存储过程开始处将set xact_abort off,那么当有错误发生时,系统不会讲xact_state()置为-1,那么我们在catch块中就不可以根据该函数值来判断是否需要进行rollback了,但是我们可以根据@@Trancount全局变量来判断,如果在catch块中判断出@@Trancount数值大于0,代表还有未提交的事务,既然进入catch语句块了,那么还存在未提交的事务,该事务应该是需要rollback的,但是这种方法在某些情况下可能判断的不准确。
推荐的方法还是将set xact_abort on,然后在catch中判断xact_state()的值来判断是否需要Rollback操作。
下面再来看个例子,在实践中不断熟悉。
这个例子就是,如果插入重复的数据给出提示信息并返回

- -- 判断要创建的存储过程名是否存在
- if Exists(Select name From sysobjects Where name = 'P_InsertUser' And type = 'P')
- -- 删除存储过程
- Drop Procedure dbo.P_InsertUser
- Go

- USE [StoreTest]
- GO
- create Procedure [dbo].[P_InsertUser]
- @UserName varchar(100),
- @UserPwd varchar(100)
- AS
- Begin
- Set NOCOUNT ON;
- Set XACT_ABORT ON; --这句话非常重要
- Begin try
- if(isnull(@UserName,'')='')
begin- print 'UserName is empty';
return;
end- declare @iCount int;
- set @iCount = 0;
- select @iCount = Count(1) from userinfo with(nolock) where username=@UserName;
- if( @iCount > 0 )
- begin
- print 'the current name already exist';
- return
- end
- Begin Tran --开始事务,事务中不能有return语句
- --insert
- insert into userinfo(
- username
- ,userpwd
- ,RegisterTime
- )
- values(
- @UserName,
- @UserPwd,
- getdate()
- )
- Commit Tran --提交事务
- end try
- begin catch
- --在此可以使用xact_state()来判断是否有不可提交的事务,不可提交的事务
- --表示在事务内部发生错误了。Xact_state()有三种值:-1.事务不可提交;
- --1.事务可提交;0.表示没有事务,此时commit或者rollback会报错。
- if xact_state()=-1
- begin
- rollback tran; --事务回滚
- SELECT ERROR_NUMBER() AS ErrorNumber,
- ERROR_MESSAGE() AS ErrorMessage;
- end
- end catch
- Set XACT_ABORT off;
- End
- --调用存储过程
- exec [P_InsertUser] '','admin'
- select * from userinfo
- GO


- 事务的东西很多,这里希望能起到抛砖引玉的效果。
- 这里附近一下c#使用事务的语法,概念是一样是,只不过是用c#实现的。
http://msdn.microsoft.com/en-us/library/system.data.sqlclient.sqltransaction(v=vs.110).aspx

- using (SqlConnection conn = new SqlConnection(Connstring)
- {
- conn .Open();
- // Start a local transaction.
- SqlTransaction sqlTran = conn .BeginTransaction();
- // Enlist a command in the current transaction.
- SqlCommand command = conn .CreateCommand();
- //begin transaction
- command.Transaction = sqlTran;
- try
- {
- // Execute two separate commands.
- command.CommandText ="xxxxx";
- command.ExecuteNonQuery();
- command.CommandText ="yyyyyy";
- command.ExecuteNonQuery();
- // Commit the transaction.
- sqlTran.Commit();
- }
- catch (Exception ex)
- {
- // Handle the exception if the transaction fails to commit.
- lblMsg.Text = ex.Message;
- try
- {
- // Attempt to roll back the transaction.
- sqlTran.Rollback();
- }
- catch (Exception exRollback)
- {
- // Throws an InvalidOperationException if the connection
- // is closed or the transaction has already been rolled
- // back on the server.
- lblMsg.Text = exRollback.Message;
- }
- }
- }

- 未完待续......
SQL 存储过程入门(事务)(四)的更多相关文章
- SQL 存储过程入门(事务)
本篇我们来讲一下事务处理技术. 为什么要使用事务呢,事务有什么用呢,举个例子. 假设我们现在有个业务,当做成功某件事情的时候要向2张表中插入数据,A表,B表,我们插入的顺序是先插入A,再插入B表,如果 ...
- SQL 存储过程 触发器 事务
一.存储过程 存储过程:就像函数一样的会保存在:数据库中-->可编程性 --> 存储过程 创建存储过程:create proc JiaFa --存储关键字proc @a int, ...
- SQL 存储过程入门(五)
好久没来博客园留下点东西,忙到找工作,最近把工作落实了. 最近公司的业务都是存储过程开发,发现去维护起来相当困难. 由于是维护项目,产品很久前都发布了,然而有一些修改,让我们去修改现在的逻辑,去看懂人 ...
- SQL 存储过程中事务回滚
在事务语句最前面加上 set xact_abort on GO SET QUOTED_IDENTIFIER OFF GO ALTER PROCEDURE [dbo].[test] @a int, @b ...
- SQL 存储过程加事务的使用
create proc USP_CUTTING_TATABLET_PULL_FINISH ( @name NVARCHAR(20) ) as SET XACT_ABORT ON--设置全盘回滚 BEG ...
- 06 数据库入门学习-视图、sql注入、事务、存储过程
一.视图 1.什么是视图 视图本质是一张虚拟的表 2.为什么要用 为了原表的安全 只要有两大功能 1.隐藏部分数据,开放指定数据 2.视图可以将查询结果保存,减少sql语句的次数 特点: 1.视图使用 ...
- 【转载】 mybatis入门系列四之动态SQL
mybatis 详解(五)------动态SQL 目录 1.动态SQL:if 语句 2.动态SQL:if+where 语句 3.动态SQL:if+set 语句 4.动态SQL:choose(when, ...
- sql之视图、触发器、函数、存储过程、事务
视图 # 视图也是一张表,但在data文件里只有表结构,没有表数据 # 不建议使用,扩展性差,程序需改变时,依赖的视图也要改变 # 视图牵涉到多张表时,视图中的记录不能修改. create view ...
- sql server 存储过程,事务
1.存储过程,事务 CREATE PROCEDURE Proc_ceshi @id int, ), @returnval int output AS BEGIN SET NOCOUNT ON; Set ...
随机推荐
- 团队项目——打地鼠游戏(SPEC)系统性能评估测试
1.SPEC测试的目标: 本轮测试的目的是测试打地鼠游戏的需求以及确保每个需求都能得到满足的方法.编写此需求说明书是为了使用户和开发人员对所开发的系统有一致的理解.通过阅读此说明书,开发人员可以了解当 ...
- 使用SharePoint CSOM 编写高效的程序
上一篇文章中简单的介绍了使用CSOM进行编程.今天主要讲一下CSOM使用中一些小技巧,可以让你的程序运行的更快. 单独加载某些属性 在上文中的例子,需要返回Web对象信息的时候,我们使用了如下的代码: ...
- Zabbix学习笔记一:基本安装与配置
1.下载安装 http://120.52.73.43/tenet.dl.sourceforge.net/project/zabbix/ZABBIX%20Latest%20Stable/3.0.1/za ...
- iOS学习笔记-精华整理
iOS学习笔记总结整理 一.内存管理情况 1- autorelease,当用户的代码在持续运行时,自动释放池是不会被销毁的,这段时间内用户可以安全地使用自动释放的对象.当用户的代码运行告一段 落,开始 ...
- 安装Vmware workstation虚拟机(含软件和注册码)
1:虚拟机的安装步骤,包含软件包和注册码.本博客所使用的虚拟机版本是vmware-workstation_11.1.0版本,注意不是最新版本, 软件包:http://pan.baidu.com/s/1 ...
- Android 学习之显式激活与隐式激活Activity
在res界面里面有两个布局文件activity_main和acivity_two
- Spring第一天
Spring框架 1.1:了解Spring Spring的核心是提供了一个容器,主要通过 BeanFactory(接口)来创建和管理对象,一般我们用它的子类ApplicationContext 来创建 ...
- android: permission和uses-permission
首先,先看一下permission定义的格式: <permission android:description="string resource" android:icon= ...
- 什么是automatic variable?
看代码符号$?搞不清楚是什么? 看代码. $share = Get-WmiObject -Class Win32_Share -ComputerName $Server.name -Credent ...
- android对象关系映射框架ormlite之一对多(OneToMany)
前两天,用ormlite对单张表进行了基本的操作,但是,我们知道通常情况对于单张表格进行操作在实际情况中很前两天不现实,那么ormlite能否像Hibenate那样实现多张表之间的一对多,多对多(即O ...