事务格式如下:
  1、开启事务: begin tran
  2、提交事务:commit tran
  3、回滚事务:rollback tran
  
  判断事务是提交还是应该回滚有两种方式,一是全局变量 @@error,二是 begin try……end try begin catch……end catch
一、@@error
  @@error作为全局变量,保存着每次执行SQL语句的状态值,当值为0时表示执行正常,否则为错误,因此可以定义一个变量,在每次执行一条SQL语句时都要将该变量的值加上@@error的值,到最后判断该变量值若为0则执行正常可以提交事务,否则回滚。
  我们通过银行转账的例子来看一下 @@error 的用法
  1、创建相关表 Bill

create table Bill
(
ID uniqueidentifier primary key,
name varchar(20),
Balance money
)

2、插入数据并查询数据

insert into Bill values (NEWID(),'小龙',4000),(newid(),'小李',2000)

select * from Bill

如图:

  3、创建存储过程

Create proc Transfer
  @IDFrom uniqueidentifier,@IDTo uniqueidentifier,@Amount money  --参数
as
  declare @errs int =0  --判断出现错误SQL语句的变量,设值为 0
  if (select COUNT(1) from Bill where ID =@IDFrom )>0 and (select COUNT(1) from Bill where ID =@IDTo)>0   --判断转入转出账户是否存在  
      and (select Balance from Bill where ID =@IDFrom)>@Amount+1000  --要求转出账户转出后余额要大于 1000 元
    begin
      begin tran  --开启事务
      update Bill set Balance=Balance -@Amount where ID =@IDFrom
      set @errs =@errs +@@ERROR   --若上一句 update 语句有问题则令 @errs +1 
      update Bill set Balance=Balance +@Amount where ID =@IDTo
      set @errs =@errs +@@ERROR   --若上一句 update 语句有问题则令 @errs +1 
      if @errs =0  --当两条 update 语句无问题时 提交 事务
        begin
          commit tran
          print '转账成功'
        end
      else  --否则 回滚
        rollback tran
    end
  else
    print '转账失败'

:由于全局变量@@ERROR 存储的永远只是上一句SQL语句执行的情况,当执行正常时返回0,否则返回不等于0的值,而再执行一句SQL语句时,执行状态会覆盖上一次的状态吗,因此要用@errs来累计每一句SQL语句的状态值。

    4、执行存储过程,小龙向小李转账 2000

exec Transfer '9FCCA201-8807-4ADA-A482-C63FD8003E54', 'C3A1A21D-3277-4962-AE97-E4C5F8B7F2EB', 2000

     显示转账成功,如图

运行  select * from Bill

如图:

再运行    exec Transfer '9FCCA201-8807-4ADA-A482-C63FD8003E54', 'C3A1A21D-3277-4962-AE97-E4C5F8B7F2EB', 2000

此时由于小龙余额只有2000,达不到要求的转账2000后余额要大于1000的要求,所以转账失败,如图:

二、begin try……end try     begin catch……end catch  方式。

在begin try……end try写事务中的执行代码并在最后提交事务,当执行SQL语句出现问题时,会跳转到  begin catch……end catch 中,因此可以在 begin catch……end catch 中写回滚事务。

  我们以订单号的例子来看一下 begin try……end try 的用法(每次获取最大的号,并+1)

    1、建表并插入数据如下:

create table Orders
(
  name varchar(10),
  OrdersID int,
)
insert into Orders values('订单号',1000)

  2、新建查询,代码如下

  begin tran 
  declare @orderID int
  declare @n int=0
  begin try
    while @n<20  --循环20次
    begin
      select @orderID =max(Orders .OrdersID) from Orders with(xlock,rowlock)
      set @orderID=@orderID +1
      print '链接 1 :'+CONVERT(nvarchar(20),@orderID)
      waitfor delay '00:00:01'
      insert into Orders values('链接 1 ',@orderID) 
      set @n=@n+1
    end
    commit tran
  end try
  begin catch
    rollback tran
  end catch

  select * from Orders

  3、再次新建查询,代码与 2 相似,将'链接 1 '改为'链接 2 ' 代码如下

begin tran 
  declare @orderID int
  declare @n int=0
  begin try
    while @n<20  --循环20次
    begin
      select @orderID =max(Orders .OrdersID) from Orders with(xlock,rowlock)
      set @orderID=@orderID +1
      print '链接 2 :'+CONVERT(nvarchar(20),@orderID)
      waitfor delay '00:00:01'
      insert into Orders values('链接 2 ',@orderID) 
      set @n=@n+1
    end
    commit tran
  end try
  begin catch
    rollback tran
  end catch

  select * from Orders

  4、再次新建查询,将'链接 1 '改为'链接 3 ' 。当然新建查询越多越好,以模拟高并发。并将3个新建查询中代码执行,结果如图,发现号码并没有丢失。

附:锁的兼容性表

现有的授权模式

链接: http://www.cnblogs.com/guanjie20/archive/2013/02/17/2914488.html

SqlServer——事务一编程进阶(SqlServer技术内幕 T-SQL程序设计 第九章的更多相关文章

  1. 《[MySQL技术内幕:SQL编程》读书笔记

    <[MySQL技术内幕:SQL编程>读书笔记 2019年3月31日23:12:11 严禁转载!!! <MySQL技术内幕:SQL编程>这本书是我比较喜欢的一位国内作者姜承尧, ...

  2. [推荐]ORACLE PL/SQL编程详解之一:PL/SQL 程序设计简介(千里之行,始于足下)

    原文:[推荐]ORACLE PL/SQL编程详解之一:PL/SQL 程序设计简介(千里之行,始于足下) [推荐]ORACLE PL/SQL编程详解之一: PL/SQL 程序设计简介(千里之行,始于足下 ...

  3. JavaScript DOM编程艺术-学习笔记(第八章、第九章)

    第八章 1.小知识点: ①某些浏览器要根据DOCTYPE 来决定页面的呈现模式(标准模式 / 怪异模式--也称兼容模式): 兼容模式意味着浏览器要模仿老一辈的浏览器的怪异行为,来让老站点得到运行,并让 ...

  4. SqlServer——事务一进阶之锁的概念(SqlServer技术内幕 T-SQL程序设计 第九章)

         一.事务的概念及ACID特性 对于单独一条SQL语句,数据库会隐式的将其作为事务,即该SQL语句要么执行成功,要么失败(相当于不执行),而我们通常说的事务就是将多条SQL语句放在 begin ...

  5. MySQL技术内幕:SQL编程 第2章 数据类型 读书笔记

    2.1 类型属性 2.1.1 UNSIGNED 数字无符号化, INT的值 -2147483648 ~ 2147483647  INT UNSIGNED的值 0 ~ 4294967295 int a ...

  6. 《Spring技术内幕》笔记-第四章 Spring MVC与web环境

    ​上下文在web容器中的启动 1,IoC容器的启动过程 IoC的启动过程就是建立上下文的过程.该上下文是与ServletContext相伴.在Spring中存在一个核心控制分发器,Dispatcher ...

  7. 《java编程思想》P160-P180(第八章部分+第九章部分)

    1.什么是多态? 多态的定义:指允许不同类的对象对同一消息做出响应.即同一消息可以根据发送对象的不同而采用多种不同的行为方式.(发送消息就是函数调用) 现实中,关于多态的例子不胜枚举.比方说按下 F1 ...

  8. 元类理解与元类编程 《Python3网络爬虫开发》中第九章代理的使用代码Crawler中代码的理解

    __new__与__init__的理解 __new__()方法是在创建实例之前被调用的,它的作用是创建一个实例,然后返回该实例对象,它是一个静态方法. __init__() 当实例被创建完成之后被调用 ...

  9. SQL Server技术内幕笔记合集

    SQL Server技术内幕笔记合集 发这一篇文章主要是方便大家找到我的笔记入口,方便大家o(∩_∩)o Microsoft SQL Server 6.5 技术内幕 笔记http://www.cnbl ...

随机推荐

  1. 【转】数据库获得当前时间getdate()

    CONVERT(nvarchar(10),count_time,121): CONVERT为日期转换函数,一般就是在时间类型 (datetime,smalldatetime)与字符串类型(nchar, ...

  2. Boostarp-响应式

    一.响应式 响应式介绍 - 响应式布局是什么? 同一个网页在不同的终端上呈现不同的布局等 - 响应式怎么实现的? 1. CSS3 media query 媒体查询 2. JS去控制网页的布局和样式等 ...

  3. js状态模式

    状态模式,当一个对象的内在状态改变时允许改变其行为,这个对象看起来是改变了其类. 状态模式主要解决的是当控制一个对象状态转换的条件表达式过于复杂时的情况.把状态的判断逻辑转移到表示不同状态的一系列类当 ...

  4. asp.net JS取值

    <script> $(function () { $.post("RegisterNew.aspx", { "ddlprovince": $(&qu ...

  5. python中的excel操作

    一. Excel在python中的应用 存测试数据 有的时候大批量的数据,我们需要存到数据库中,在测试的时候才能用到.测试的时候就从数据库中读取出来.这点是非常重要的! 存测试结果 二. Excel中 ...

  6. 强制关闭iPhone iPad AppleWatch MacOS

    iPhone/iPad 强制关闭APP:按住Power(电源键),出现关机界面松开,按住Home键9秒左右. 强制重启/关机:同时按住Power和Home键几秒后重启,出现logo时松开Home继续按 ...

  7. MinGW main()

    MinGW没有wmain入口函数,为了获取宽字符的参数,可以用系统API函数GetCommandLineW. main.cpp #include <iostream> #include & ...

  8. MySQL管理

    http://www.yiibai.com/mysql/administration.html 在本节中,您将学习有关MySQL管理教程,包括MySQL服务器启动和关闭,MySQL服务器安全性,MyS ...

  9. IE9 placeholder 不兼容的解决

    坑爹的IE9-,真的是够够的了,不过公司不要求兼容这个玩意了,自己觉得兼容这个鬼还是挺有挑战性的,自己也碰到不少难题,一个个解决. css: .placeholderColor { color : # ...

  10. Codeforces Round #254(div2)B

    就是看无向图有几个连通块,答案就是2n-num. 范围很小,就用矩阵来存图减少代码量. #include<iostream> #include<cstdio> #include ...