mysql乐观锁怎么写 炒股软件文字出现乱码怎么办?

正确理解MYSQL的幻读一、定义
1、幻读MYSQL官方叫法是Phantom Rows,意为鬼影行或者幻影行,请看官方定义:
The so-called phantom problem occurs within a transaction when the same query produces different sets of rows at different times. For example, if a [ SELECT ] is executed twice, but returns a row the second time that was not returned the first time, the row is a “phantom” row.
翻译一下:
所谓的幻影行问题是指,在同一个事务中,同样的查询语句执行多次,得到了不同的行结果集 。
例如,如果同一个SELECT语句执行了两次,第二次执行的时候比第一次执行时多出一行,则该行就是所谓的幻影行 。
2、幻读与不可重复读的区别
从官方的定义来看,幻读的定义侧重于多条记录,就是记录条数的变化,而不可重复读侧重于单条记录数据的变化,这样区分原因在于解决幻读需要范围锁 , 解决不可重复读只需要单条记录加锁
二、InnoDB的REPEATABLE READ级别
InnoDB支持由SQL1992标准描述的所有四个事务隔离级别 , 默认隔离级别是 REPEATABLE READ 。
1、快照读:
在RR模式下,第一次读取会建立快照,后续查询会读取快照 。
这意味着,如果在同一事务中发出多个普通[ SELECT ](非锁定)语句,则这些 [ SELECT ]语句的结果也是一致的 。
2、[locking reads](锁定读?。?又叫当前读)
[ SELECT ]语句中使用 FOR UPDATE 或 FOR SHARE
3、行锁
在RR模式下,使用当前读以及 [ UPDATE ]和 [ DELETE ]语句会对数据记录加行锁,锁定范围取决于该语句使用的是具有唯一搜索条件的唯一索引还是范围类型搜索条件 。
三、InnoDB的READ COMMITTED级别
1、在RC模式下,每次读取都会刷新快照,因此不能保证可重复读
2、在RC模式下,使用当前读以及 [ UPDATE ]和 [ DELETE ]语句会对数据记录加行锁 , 但是不会加范围锁,间隙锁定仅用于外键约束检查和重复键检查 。
3、由于禁用了间隙锁定,因此可能会产生幻影行问题,因为其他会话可以在间隙中插入新行 。
4、 对于[ UPDATE ]或 [ DELETE ]语句,InnoDB 仅对其更新或删除的行持有锁 。MySQL评估 WHERE 条件后,将释放不匹配行的记录锁。这大大降低了死锁的可能性 , 但是仍然可以发生 。
5、对于[ UPDATE ]语句,如果某行已被锁定,则 InnoDB执行“半一致”读取,将最新提交版本的数据返回给MySQL,以便MySQL可以确定该行是否符合WHERE 条件 。如果该行匹配(必须更新),则MySQL会再次读取该行 , 这一次 InnoDB 会将其锁定或等待获取锁 。
6、注意
从MySQL 8.0.22开始 , DML操作(增删改 , 通过联接列表或子查询)从MySQL授权表中读取数据,但不对其进行修改 , 无论隔离级别如何,都不会在MySQL授权表上获得读取锁 。
有关更多信息,请参见Grant Table Concurrency。
四、乐观锁与悲观锁
1、乐观锁
在UPDATE的WHERE子句中加入版本信息来确定修改是否生效
使用乐观锁时仍然需要非常谨慎,因为RR是可重复读的,在UPDATE之前读取版本号,应该使用[当前读],不能使用[快照读]
2、悲观锁
在UPDATE执行前 , SELECT后面加上FOR UPDATE来给记录加锁 , 保证记录在UPDATE前不被修改 。SELECT ... FOR UPDATE是加上了X锁,也可以通过SELECT ... LOCK IN SHARE MODE加上S锁,来防止其他事务对该行的修改 。
3、无论是乐观锁还是悲观锁,使用的思想都是一致的,那就是当前读 。乐观锁利用当前读判断是否是最新版本,悲观锁利用当前读锁定行 。
五、总结

推荐阅读