为什么需要锁(并发控制)?
来源:互联网 发布:unity3d游戏开发基础 编辑:程序博客网 时间:2024/06/09 20:41
为什么需要锁(并发控制)?
在多用户环境中,在同一时间可能会有多个用户更新相同的记录,这会产生冲突。这就是著名的并发性问题。
典型的冲突有:
l 丢失更新:一个事务的更新覆盖了其它事务的更新结果,就是所谓的更新丢失。例如:用户A把值从6改为2,用户B把值从2改为6,则用户A丢失了他的更新。
l 脏读:当一个事务读取其它完成一半事务的记录时,就会发生脏读取。例如:用户A,B看到的值都是6,用户B把值改为2,用户A读到的值仍为6。
为了解决这些并发带来的问题。 我们需要引入并发控制机制。
并发控制机制
悲观锁:假定会发生并发冲突,屏蔽一切可能违反数据完整性的操作。[1]
乐观锁:假设不会发生并发冲突,只在提交操作时检查是否违反数据完整性。[1] 乐观锁不能解决脏读的问题。
乐观锁应用
1. 使用自增长的整数表示数据版本号。更新时检查版本号是否一致,比如数据库中数据版本为6,更新提交时version=6+1,使用该version值(=7)与数据库version+1(=7)作比较,如果相等,则可以更新,如果不等则有可能其他程序已更新该记录,所以返回错误。
2. 使用时间戳来实现.
注:对于以上两种方式,Hibernate自带实现方式:在使用乐观锁的字段前加annotation: @Version, Hibernate在更新时自动校验该字段。
悲观锁应用
需要使用数据库的锁机制,比如SQL SERVER 的TABLOCKX(排它表锁) 此选项被选中时,SQL Server 将在整个表上置排它锁直至该命令或事务结束。这将防止其他进程读取或修改表中的数据。
SqlServer中使用
Begin Tran
select top 1 @TrainNo=T_NO
from Train_ticket with (UPDLOCK) where S_Flag=0
update Train_ticket
set T_Name=user,
T_Time=getdate(),
S_Flag=1
where T_NO=@TrainNo
commit
我们在查询的时候使用了with (UPDLOCK)选项,在查询记录的时候我们就对记录加上了更新锁,表示我们即将对此记录进行更新. 注意更新锁和共享锁是不冲突的,也就是其他用户还可以查询此表的内容,但是和更新锁和排它锁是冲突的.所以其他的更新用户就会阻塞.
结论
在实际生产环境里边,如果并发量不大且不允许脏读,可以使用悲观锁解决并发问题;但如果系统的并发非常大的话,悲观锁定会带来非常大的性能问题,所以我们就要选择乐观锁定的方法.
- 为什么需要锁(并发控制)?
- 从头认识java-18.1 为什么需要并发?
- 版本控制之为什么需要版本控制器
- C++ 11 并发控制(锁)
- 并发控制__锁(mysql)
- 锁和并发控制
- (十一)并发控制
- 为什么单例对象的并发调用需要同步?
- 为什么单例对象的并发调用需要同步?
- 为什么单例对象的并发调用需要同步?
- 为什么单例对象的并发调用需要同步?
- C#为什么多线程控制winform需要用委托?
- C#为什么多线程控制winform需要用委托?
- C#为什么多线程控制winform需要用委托?
- 为什么需要框架(java)
- 为什么需要使用并发编程?什么时候适合使用并发编程技术?
- innodb并发控制mvcc(多版本并发控制)
- 并发控制——“锁”
- 常见三维重建算法及其实现(含源码下载链接)
- 前端优化-渐进式图片的制作
- 欢迎使用CSDN-markdown编辑器
- Fragment一些方法介绍
- 在Visual C++ 中使用内联汇编
- 为什么需要锁(并发控制)?
- UVA101
- xpath 解析一
- MySQL应用架构优化-实时数据处理
- json字符串与json对象转换
- 页面跳页面传值问题
- HashMap源码透一透
- MySql 外键约束 之CASCADE、SET NULL、RESTRICT、空等类型分析和作用解读
- 蒙哥马利算法详解