gpt4 book ai didi

java - JPA/Eclipselink @Cache 过期被忽略

转载 作者:行者123 更新时间:2023-11-30 04:20:22 30 4
gpt4 key购买 nike

我目前正在评估 JPA/Eclipselink 作为我们当前非常丑陋的 jdbc 数据库访问的替代方案。情况如下:

多个客户端访问同一数据库,有时会编辑相同的数据。这些数据应该定期刷新,并且只缓存很短的一段时间。根据我的理解,@Cache 注释应该能够做到这一点。

以下代码基于Vogella上的jpa/eclipselink

@Entity
@Cache(expiry = 100)
public class Todo {
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
private Long id;
private String summary;
private String description;
... getter/setter/toString omited
}

public static void main(String[] args) throws InterruptedException {
EntityManagerFactory factory =
Persistence.createEntityManagerFactory("mysql");
EntityManager em = factory.createEntityManager();

TypedQuery<Todo> q = em.createQuery("SELECT t FROM Todo t", Todo.class);

List<Todo> todoList = q.getResultList();
System.out.println("Query 1 read");
for (int i = 0; i < todoList.size(); i++) {
System.out.println("1: " + todoList.get(i));
}

Thread.sleep(30000);

TypedQuery<Todo> q2 =
em.createQuery("SELECT t FROM Todo t", Todo.class);
// q2.setHint("javax.persistence.cache.storeMode", "REFRESH");

List<Todo> todoList2 = q2.getResultList();
System.out.println("Query 2 read");

for (int i = 0; i < todoList.size(); i++) {
System.out.println("1: " + todoList.get(i));
System.out.println("2: " + todoList2.get(i));
System.out.println(todoList.get(i) == todoList2.get(i));
}
}

<?xml version="1.0" encoding="UTF-8"?>
<persistence version="2.0"
xmlns="http://java.sun.com/xml/ns/persistence" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://java.sun.com/xml/ns/persistence http://java.sun.com/xml/ns/persistence/persistence_2_0.xsd">


<persistence-unit name="mysql" transaction-type="RESOURCE_LOCAL">
<provider>org.eclipse.persistence.jpa.PersistenceProvider</provider>
<exclude-unlisted-classes>false</exclude-unlisted-classes>
<properties>
<property name="javax.persistence.jdbc.password" value="" />
<property name="javax.persistence.jdbc.user" value="root" />
<property name="javax.persistence.jdbc.driver" value="com.mysql.jdbc.Driver" />
<property name="javax.persistence.jdbc.url" value="jdbc:mysql://localhost:3306/test" />
<property name="javax.persistence.ddl-generation" value="create-tables" />
<property name="javax.persistence.logging.level" value="ALL" />
<property name="eclipselink.logging.level" value="ALL"/>
</properties>
</persistence-unit>
</persistence>

运行程序会产生以下输出:

...
[EL Finest]: query: 2013-06-19 14:30:55.897--UnitOfWork(31211079)--Thread(Thread[main,5,main])--Execute query ReadAllQuery(referenceClass=Todo sql="SELECT ID, DESCRIPTION, SUMMARY FROM TODO")
[EL Finest]: connection: 2013-06-19 14:30:55.907--ServerSession(7427424)--Connection(7633596)--Thread(Thread[main,5,main])--Connection acquired from connection pool [default].
[EL Fine]: sql: 2013-06-19 14:30:55.907--ServerSession(7427424)--Connection(7633596)--Thread(Thread[main,5,main])--SELECT ID, DESCRIPTION, SUMMARY FROM TODO
[EL Finest]: connection: 2013-06-19 14:30:55.924--ServerSession(7427424)--Connection(7633596)--Thread(Thread[main,5,main])--Connection released to connection pool [default].
Query 1 read
1: Todo [id=1, summary=s1, description=d1]
1: Todo [id=2, summary=qwet, description=d2]
[EL Finest]: query: 2013-06-19 14:31:25.932--UnitOfWork(31211079)--Thread(Thread[main,5,main])--Execute query ReadAllQuery(referenceClass=Todo sql="SELECT ID, DESCRIPTION, SUMMARY FROM TODO")
[EL Finest]: connection: 2013-06-19 14:31:25.932--ServerSession(7427424)--Connection(7633596)--Thread(Thread[main,5,main])--Connection acquired from connection pool [default].
[EL Fine]: sql: 2013-06-19 14:31:25.932--ServerSession(7427424)--Connection(7633596)--Thread(Thread[main,5,main])--SELECT ID, DESCRIPTION, SUMMARY FROM TODO
[EL Finest]: connection: 2013-06-19 14:31:25.934--ServerSession(7427424)--Connection(7633596)--Thread(Thread[main,5,main])--Connection released to connection pool [default].
Query 2 read
1: Todo [id=1, summary=s1, description=d1]
2: Todo [id=1, summary=s1, description=d1]
true
1: Todo [id=2, summary=qwet, description=d2]
2: Todo [id=2, summary=qwet, description=d2]
true

当线程 hibernate 时,我更改数据库中的值。我希望从数据库中获取新值,因为缓存中的值应该已经过期。看起来代码只是忽略了 @Cache 注释。我还尝试了各种其他设置,例如 type=CacheType.NONE 和alwaysRefresh=true,但没有任何改变。

当我添加 storeMode-QueryHint 时,查询始终刷新结果。这并不完全是我想要的,似乎我需要将它添加到每个查询中,这很容易出错。但还是很奇怪,Cache注解被忽略了。

我还尝试使用 DescriptorCustomizer,但它也没有效果,尽管已使用(使用断点测试)。

public void customize(ClassDescriptor descriptor) {
descriptor.alwaysRefreshCache();
descriptor.alwaysRefreshCacheOnRemote();
descriptor.disableCacheHits();
descriptor.disableCacheHitsOnRemote();
}

更新:

对我正在开发的系统的一些话。它是读/写我们的主数据的模块。因此,例如,同一用户的用户对象在内存中出现两次是错误的。另外,在我测试了 @ReadOnly 注释之后,我注意到 @Cache 注释的隔离参数似乎可以工作,但仍然没有关于过期、alwaysrefresh 等的信息。

最佳答案

您正在使用相同的 EntityManager 实例。缓存设置适用于共享缓存,而 EntityManager 需要将同一对象保留在内存中,直到它被关闭或清除 - 否则简单的查找操作可能会删除正在运行的事务中的更改。获取一个新的 EntityManager 或清除现有的 EntityManager 将导致 EntityManager 根据需要转到共享缓存和/或数据库。

如果您确实需要在当前 EM 上下文中刷新对象,则必须显式调用 em.refresh 或使用查询提示,以便它知道清除数据库中的任何现有更改。

每个 EM 代表一个事务上下文,并且由于每个线程都应该有自己的 EntityManager,因此应用程序中已经存在实体的多个副本。如果您依赖于从 EntityManager 读取的特定实例始终反射(reflect)当前更改,您将会遇到麻烦 - 它只能真正反射(reflect)读取时数据库中的内容。这就是为什么在必要时合并更改很重要,并且在应用程序中缓存对象可能不是一个好主意 - 根据需要从 EntityManager 访问它们。

如果您想要该对象并且不进行更改,请将其标记为只读,这将从共享缓存而不是 EM 中提取它,从而反射(reflect)缓存设置:http://wiki.eclipse.org/EclipseLink/UserGuide/JPA/Basic_JPA_Development/Caching/Shared_and_Isolated#Read-Only_Entities 。如果实体缓存在共享缓存中,这将是您返回的实例,并且所有 EM 将返回相同的对象实例 - 由您的应用程序来管理实体本身的并发问题。

关于java - JPA/Eclipselink @Cache 过期被忽略,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/17191680/

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