MySQL的insert on duplicate 逻辑是,先插入主键,再插唯一键,如果唯一键发生冲突会出现加next-key锁,然后进行一次回滚(要回滚写入到主键的记录),此时刚刚插入的遗留在主键记录的锁(MySQL的隐式锁机制)退化成了gap锁(因为主键上的记录不再需要了),锁的范围就退化成主键的最后到无穷大,在批量更新场景下,也许会有另外一个行数据插入到最后,还会有一把插入意向锁,此时死锁。
无论是死锁还是锁放大,根源在于锁的范围扩大了。为什么范围扩大了,因为sql语句种,无论是replace还是insert 之类,都会在插入或者更新前,先去判断下是不是这个b树上有唯一值存在。因为b树里存在大量的被标记为删除但是还未真正删除的记录,在检查唯一值的时候,就需要把这些全部先锁住,以及这些前后以及间隙都锁住,造成了锁范围放大。所以建议大家还是最好不要用这个语法。如果一定要用insert on duplicate,避免死锁的解法就是一次只更新一行。