7

我有一个谷歌番石榴缓存,它从数据库加载数据并使用主键缓存它。我最终从数据库创建的对象是不可变的,构建一个对象需要访问多个表。在以下情况下会发生什么:

  • 线程 1:调用 cache.load(10) 并根据值为 10 的数据库主键填充缓存
  • 线程 2:使用主键 10 更新数据库行,因此它调用 cache.invalidate(10) 在 cache.load(10) 完成之前调用 invalidate。

在 load(x) 执行时调用 invalidate(x) 时 Guava Loading Cache 会做什么?

4

2 回答 2

6

当前在Javadoc中指定“在加载完成之前,不会修改与此缓存关联的可观察状态”。加载的语义进一步指定为“加载Cache.asMap().putIfAbsent完成后使用新加载的值添加到缓存中”。

您还可以阅读代码以查看调用 invalidate 或 remove 时忽略加载条目的位置。

于 2012-10-15T08:12:42.233 回答
3

你可以有两种情况:

  • 线程 1 首先到达实际加载点(LocalCache.Segment.lockedGetOrLoad()在 13.0.1 中),并获得了段锁:在这种情况下,加载完成,释放锁并将计算值返回给调用者,但它将是线程 2 在运行 ( LocalCache.Segment.remove()) 并且可以获取锁时使其失效。

  • 线程 2 在线程 1 实际开始加载之前获得了锁:失效实际上并没有做任何事情,因为条目还没有,然后线程 1 加载最新的值。

于 2012-10-15T08:13:27.007 回答