监控(Watch)
悲观锁:
- 很悲观,认为什么时候都会出问题,无论做什么都会加锁
乐观锁:
- 很乐观,认为什么时候都不会出问题,所以不会上锁。更新数据的时候去判断一下,在此期问是否有人修改过这个数据。
- 获取version
- 更新的时候比较version
Redis 的监视测试
正常执行
127.0.0.1:6379> set money 100
OK
127.0.0.1:6379> set out 0
OK
127.0.0.1:6379> watch money # 监事 money 对象
OK
127.0.0.1:6379> multi # 事务正常结束,数据期间没有发生变动,这个时候就正常执行成功
OK
127.0.0.1:6379> decrby money 20
QUEUED
127.0.0.1:6379> incrby out 20
QUEUED
127.0.0.1:6379> exec
1) (integer) 80
2) (integer) 20
测试多线程修改值
127.0.0.1:6379> set money 100
OK
127.0.0.1:6379> set out 0
OK
开启两个客户端
客户端1:
127.0.0.1:6379> watch money # 监事 money
OK
127.0.0.1:6379> multi
OK
127.0.0.1:6379> decrby money 10
QUEUED
127.0.0.1:6379> incrby out 10
QUEUED
此时执行客户端2:
127.0.0.1:6379> get money
"100"
127.0.0.1:6379> set money 1000
OK
回来执行客户端1:
127.0.0.1:6379> exec # 执行之前,另外一个线程修改了值。这个时候就会导致事务执行失败!
(nil)
客户端1发现事务执行失败,就先解锁 ,再监视
127.0.0.1:6379> unwatch # 如果发现事务执行失败,就先解锁
OK
127.0.0.1:6379> watch money # 获取最新的值,再次监视
OK
127.0.0.1:6379> multi
OK
127.0.0.1:6379> decrby money 10
QUEUED
127.0.0.1:6379> incrby out 10
QUEUED
127.0.0.1:6379> exec # 比对监视的值有没有发生变化
1) (integer) 990
2) (integer) 10
总结:watch 监视 money 的值,如果没有发生变化,就执行成功;一旦发生变化,事务就执行失败