时间:2021-07-01 10:21:17 帮助过:3人阅读
事务:保持逻辑数据一致性与可恢复性,必不可少的利器。
锁:多用户访问同一数据库资源时,对访问的先后次序权限管理的一种机制,没有他事务或许将会一塌糊涂,不能保证数据的安全正确读写。
死锁:是数据库性能的重量级杀手之一,而死锁却是不同事务之间抢占数据资源造成的。
不懂的听上去,挺神奇的,懂的感觉我在扯淡,下面带你好好领略下他们的风采,嗅査下他们的狂骚。。
用华仔无间道中的一句来给你诠释下:去不了终点,回到原点。
举例说明:
在一个事务中,你写啦2条sql语句,一条是修改订单表状态,一条是修改库存表库存-1 。 如果在修改订单表状态的时候出错,事务能够回滚,数据将恢复到没修改之前的数据状态,下面的修改库存也就不执行,这样确保你关系逻辑的一致,安全。。
事务就是这个样子,倔脾气,要么全部执行,要么全部不执行,回到原数据状态。
书面解释:事务具有原子性,一致性,隔离性,持久性。
然而在SQL Server中事务被分为3类常见的事务:
常用语句就四个。
上面的都是心法,下面的给你来个招式,要看仔细啦。
1 ---开启事务
2 begin tran
3 --错误扑捉机制,看好啦,这里也有的。并且可以嵌套。
4 begin try
5 --语句正确
6 insert into lives (Eat,Play,Numb) values (‘猪肉‘,‘足球‘,1)
7 --Numb为int类型,出错
8 insert into lives (Eat,Play,Numb) values (‘猪肉‘,‘足球‘,‘abc‘)
9 --语句正确
10 insert into lives (Eat,Play,Numb) values (‘狗肉‘,‘篮球‘,2)
11 end try
12 begin catch
13 select Error_number() as ErrorNumber, --错误代码
14 Error_severity() as ErrorSeverity, --错误严重级别,级别小于10 try catch 捕获不到
15 Error_state() as ErrorState , --错误状态码
16 Error_Procedure() as ErrorProcedure , --出现错误的存储过程或触发器的名称。
17 Error_line() as ErrorLine, --发生错误的行号
18 Error_message() as ErrorMessage --错误的具体信息
19 if(@@trancount>0) --全局变量@@trancount,事务开启此值+1,他用来判断是有开启事务
20 rollback tran ---由于出错,这里回滚到开始,第一条语句也没有插入成功。
21 end catch
22 if(@@trancount>0)
23 commit tran --如果成功Lives表中,将会有3条数据。
24
25 --表本身为空表,ID ,Numb为int 类型,其它为nvarchar类型
26 select * from lives
---开启事务
begin tran
--错误扑捉机制,看好啦,这里也有的。并且可以嵌套。
begin try
--语句正确
insert into lives (Eat,Play,Numb) values (‘猪肉‘,‘足球‘,1)
--加入保存点
save tran pigOneIn
--Numb为int类型,出错
insert into lives (Eat,Play,Numb) values (‘猪肉‘,‘足球‘,2)
--语句正确
insert into lives (Eat,Play,Numb) values (‘狗肉‘,‘篮球‘,3)
end try
begin catch
select Error_number() as ErrorNumber, --错误代码
Error_severity() as ErrorSeverity, --错误严重级别,级别小于10 try catch 捕获不到
Error_state() as ErrorState , --错误状态码
Error_Procedure() as ErrorProcedure , --出现错误的存储过程或触发器的名称。
Error_line() as ErrorLine, --发生错误的行号
Error_message() as ErrorMessage --错误的具体信息
if(@@trancount>0) --全局变量@@trancount,事务开启此值+1,他用来判断是有开启事务
rollback tran ---由于出错,这里回滚事务到原点,第一条语句也没有插入成功。
end catch
if(@@trancount>0)
rollback tran pigOneIn --如果成功Lives表中,将会有3条数据。
--表本身为空表,ID ,Numb为int 类型,其它为nvarchar类型
select * from lives
设置 xact_abort on/off , 指定是否回滚当前事务,为on时如果当前sql出错,回滚整个事务,为off时如果sql出错回滚当前sql语句,其它语句照常运行读写数据库。
需要注意的时:xact_abort只对运行时出现的错误有用,如果sql语句存在编译时错误,那么他就失灵啦。
delete lives --清空数据
set xact_abort off
begin tran
--语句正确
insert into lives (Eat,Play,Numb) values (‘猪肉‘,‘足球‘,1)
--Numb为int类型,出错,如果1234..那个大数据换成‘132dsaf‘ xact_abort将失效
insert into lives (Eat,Play,Numb) values (‘猪肉‘,‘足球‘,12345646879783213)
--语句正确
insert into lives (Eat,Play,Numb) values (‘狗肉‘,‘篮球‘,3)
commit tran
select * from lives
为on时,结果集为空,因为运行是数据过大溢出出错,回滚整个事务。
跟着做:打开两个查询窗口,把下面的语句,分别放入2个查询窗口,在5秒内运行2个事务模块。
begin tran
update lives set play=‘羽毛球‘
waitfor delay ‘0:0:5‘
update dbo.Earth set Animal=‘老虎‘
commit tran
begin tran
update Earth set Animal=‘老虎‘
waitfor delay ‘0:0:5‘ --等待5秒执行下面的语句
update lives set play=‘羽毛球‘
commit tran
select * from lives
select * from Earth
为什么呢,下面我们看看锁,什么是锁。
在多用户都用事务同时访问同一个数据资源的情况下,就会造成以下几种数据错误。
然而锁定,就是为解决这些问题所生的,他的存在使得一个事务对他自己的数据块进行操作的时候,而另外一个事务则不能插足这些数据块。这就是所谓的锁定。
锁定从数据库系统的角度大致可以分为6种:
这些锁之间的相互兼容性,也就是,是否可以同时存在。
|
现有的授权模式 |
|
|
|
|
|
---|---|---|---|---|---|---|
请求的模式 |
IS |
S |
U |
IX |
SIX |
X |
意向共享 (IS) |
是 |
是 |
是 |
是 |
是 |
否 |
共享 (S) |
是 |
是 |
是 |
否 |
否 |
否 |
更新 (U) |
是 |
是 |
否 |
否 |
否 |
否 |
意向排他 (IX) |
是 |
否 |
否 |
是 |
否 |
否 |
意向排他共享 (SIX) |
是 |
否 |
否 |
否 |
否 |
否 |
排他 (X) |
否 |
否 |
否 |
否 |
否 |
否 |
锁兼容性具体参见:http://msdn.microsoft.com/zh-cn/library/ms186396.aspx
锁粒度和层次结构参见:http://msdn.microsoft.com/zh-cn/library/ms189849(v=sql.105).aspx
什么是死锁,为什么