gpt4 book ai didi

java - Hibernate 不驱逐查询缓存(不可重复读取问题)

转载 作者:塔克拉玛干 更新时间:2023-11-02 20:13:37 26 4
gpt4 key购买 nike

我是 java/spring/hibernate 的新手,经过几年的 .Net 编程,我真的爱上了 java。

现在我正在使用 Spring(MVC,声明式事务)和 Hibernate(3.6,作为缓存提供者 - ehCache 2.5)开发 Web 应用程序。我有一些只读和读写实体,我想使用 Hibernate 二级缓存和查询缓存进行缓存。

当我对只读实体使用缓存时一切正常。我添加了读写实体并使用 jMeter 运行了性能测试。对于读写实体,我面临着不可重复读取的问题。例如。有几个并发线程读取和写入实体表。

线程 3 获取查找值:

16:34:45,304 DEBUG [http-bio-8080-exec-3] cache.StandardQueryCache:        (StandardQueryCache.java:136) - cached query results were not up to date
16:34:45,304 DEBUG [http-bio-8080-exec-3] hibernate.SQL:(SQLStatementLogger.java:111) - select virtualdev0_.virtual_device_class_id as virtual1_45_, virtualdev0_.virtual_device_class as virtual2_45_, virtualdev0_.sitebox_id as sitebox3_45_, virtualdev0_.timestamp as timestamp45_ from virtual_device_class virtualdev0_ where virtualdev0_.sitebox_id=?

它发现缓存不是最新的并加载实体,将它们添加到二级缓存,具体化并返回......从这里持续到 16:34:45,826

同时线程 9 删除其中一个实体并更新二级缓存 + 时间戳:

16:34:45,799 DEBUG [http-bio-8080-exec-9] hibernate.SQL:(SQLStatementLogger.java:111) - delete from virtual_device_class where virtual_device_class_id=?
16:34:45,814 DEBUG [http-bio-8080-exec-9] cache.UpdateTimestampsCache:(UpdateTimestampsCache.java:95) - Invalidating space [virtual_device_class], timestamp: 5466792287494145

线程 3 继续内部管理 Activity 并最终将查询结果添加到查询缓存(注意时间戳将高于线程 9 的删除操作的时间戳):

16:34:45,826 DEBUG [http-bio-8080-exec-3] cache.StandardQueryCache:(StandardQueryCache.java:96) - caching query results in region: org.hibernate.cache.StandardQueryCache; timestamp=5466792287543296

因此此时删除的 ID 将在查询缓存中,并且查询缓存将被视为是最新的。

16:34:45,852 DEBUG [http-bio-8080-exec-9] cache.UpdateTimestampsCache:(UpdateTimestampsCache.java:122) - [virtual_device_class] last update timestamp: 5466792287494145, result set timestamp: 5466792287543296

因此,当您尝试再次进行查找时,它将在查询缓存中查找,然后开始从第二个缓存中具体化实体。

16:34:45,852 DEBUG [http-bio-8080-exec-9] cache.StandardQueryCache:(StandardQueryCache.java:140) - returning cached query results

但已删除的项目不会存在,因此将完成对数据库的查询。

16:34:45,863 DEBUG [http-bio-8080-exec-9] loader.Loader:(Loader.java:2022) - loading entity: [com.test.models.VirtualDeviceClass#0b2f363f-fbb9-4d17-8f86-af86ebb5100c]
16:34:45,873 DEBUG [http-bio-8080-exec-9] hibernate.SQL:(SQLStatementLogger.java:111) - select virtualdev0_.virtual_device_class_id as virtual1_45_0_, virtualdev0_.virtual_device_class as virtual2_45_0_, virtualdev0_.sitebox_id as sitebox3_45_0_, virtualdev0

因为我正在使用 Load 方法,所以如果在数据库中找不到实体,它会抛出异常。在我的情况下,实体很少更新,这可能会发生,这让我很担心。我对如何尝试克服这个问题没有什么想法:

a) 在数据库中将 trx 隔离级别设置为可重复读取(但是不要认为这会有帮助,因为添加到缓存逻辑是在从数据库读取数据之后发生的)b) 手动强制标准查询缓存在实体删除/更新时逐出c) 根本不使用查询缓存(尝试将大部分数据库查询路由到使用第二个缓存)

以前有人遇到过这个问题吗?

最佳答案

我已经迁移到 Hibernate 4,现在工作正常。

此问题可能与方法 SessionFactory.getQueryCache(String regionName) 中删除的同步块(synchronized block)有关

link to hibernate issue

关于java - Hibernate 不驱逐查询缓存(不可重复读取问题),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/10204876/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com