首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >问答首页 >忽略JPA / Eclipselink @Cache过期

忽略JPA / Eclipselink @Cache过期
EN

Stack Overflow用户
提问于 2013-06-19 20:51:28
回答 1查看 1.5K关注 0票数 1

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

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

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

代码语言:javascript
运行
复制
@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>

运行该程序将产生以下输出:

代码语言:javascript
运行
复制
...
[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

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

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

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

代码语言:javascript
运行
复制
public void customize(ClassDescriptor descriptor) {
        descriptor.alwaysRefreshCache();
        descriptor.alwaysRefreshCacheOnRemote();
        descriptor.disableCacheHits();
        descriptor.disableCacheHitsOnRemote();
    }

更新:

对我正在开发的系统说些话。它是读/写我们的主数据的模块。因此,例如,同一个用户在内存中拥有两次User-object是错误的。另外,在我测试了@ReadOnly注释之后,我注意到@Cache注释的isolation-参数似乎可以工作,但仍然没有关于expire、alwaysrefresh等的内容。

EN

Stack Overflow用户

发布于 2013-06-20 22:01:26

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

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

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

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

票数 2
EN
查看全部 1 条回答
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/17191680

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档