异常信息
Ccom.mysql.cj.jdbc.exceptions.MySQLTransactionRollbackException: Lock wait timeout exceeded; try restarting transaction
问题发生原因
乐观锁修改数据的时候, 数据版本号(version)已经被修改了,导致修改失败. 进行重试修改时, 每次从数据库读取出来的数据不是数据库最新版本, 导致无限次重试, 直到该事物超时自动退出
使用乐观锁(CAS)机制去更新一条记录, 从数据库查询一条数据, 当查出来数据版本号是1, 在修改数据的时候, 把数据版本号作为修改条件之一,update `table_name` set version=`2` where version=`1` , 如果修改失败, 那么该数据已经被更新过了, 重新读取数据, 进行重试, 而Mysql数据库中, 开启事物之后, 在当前事物中, 多次查询的值, 会是同一个, 即修改失败, 进行重新读数据, 得到的数据版本号是该次事物中第一次读的时候的版本号 (即数据库事物隔离级别中的可重复读, 避免脏读现象(多次读取值不一致), 类似于开启一个事物的时候, 每次读取, 会把查询的数据复制到事物空间, 当前事物读数据库的时候, 不会读表中的实际数据, 而是读事物空间的数据. 因为这个原因, 版本号始终不会跟新, 所以会一直修改失败.
问题代码表示
`[@Override](https://my.oschina.net/u/1162528)`
`@Transactional(rollbackFor = Exception.**class**) // 开始事物`
`**public** Tuple2 earn(EarnDTO earnDTO) {`
`**return** retryEarn(earnDTO);/