问题来源
使用redis做一个缓冲操作,让请求先访问到redis,而不是直接访问MySQL等数据库:
读取缓存步骤一般没有什么问题,但是一旦涉及到数据更新:数据库和缓存更新,就容易出现缓存(Redis)和数据库(MySQL)间的数据一致性问题。
不管是先写MySQL数据库,再删除Redis缓存,再更新缓存;还是先删除缓存,再写库,再更新缓存。都有可能出现数据不一致的情况。举一个例子:
1.如果删除了缓存Redis,还没有来得及写库MySQL,另一个线程就来读取,发现缓存为空,则去数据库中读取数据写入缓存,此时缓存中为脏数据。
2.如果先写了库,在删除缓存前,写库的线程宕机了,没有删除掉缓存,则也会出现数据不一致情况。
因为写和读是并发的,没法保证顺序,就会出现缓存和数据库的数据不一致的问题。
情况一:只先删除缓存
第一步删除原先的旧缓存0,使缓存为空,第二步再将新数据 1 插入数据库。但是问题在于,第一步和第二步之间,若有一个B事务去读取缓存时,发现缓存为空,B事务自然会去数据库中查数据,此时查到的数据为0,然后再把 0 放到缓存中,紧接着A事务继续进行它的第二步,将数据库的数据0改为1。这就导致了缓存为0,数据库为1,以后再有事务C来查数据时,查到的缓存一直是0,即一直为脏数据。