簡體   English   中英

JPA / Eclipselink @Cache過期被忽略

[英]JPA / Eclipselink @Cache expiry ignored

我目前正在評估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

當線程休眠時,我更改了數據庫中的值。 我希望從數據庫中獲取新值,因為緩存中的值應該已經過期。 似乎代碼只是忽略了@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批注的隔離參數似乎可以正常工作,但是對於expire,alwaysrefresh等仍然無效。

您正在使用相同的EntityManager實例。 緩存設置適用於共享緩存,而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 如果將實體緩存在共享緩存中,這將是您返回的實例,並且所有EM將返回相同的對象實例-由您的應用來管理實體本身的並發問題。

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM