【问题标题】:JPA + Hibernate + EHCache, unexpected behaviorJPA + Hibernate + EHCache,意外行为
【发布时间】:2012-07-18 19:20:01
【问题描述】:

我正在尝试将 EHCache 实现到我的工作原型中,其中我有一个 javax.persistence.Entity 表示我的数据库(MySQL,mysql-connector-java-5.1.20. jar),以 XML 形式提供给消费者。

我面临的问题是,Hibernate 似乎仍在从数据库中检索数据,即使 EHCache 将查询结果存储在内存中

我正在使用EHCache monitor查看内存中的项目数,并在缓存过期之前直接更改数据库上的数据以了解是否实际使用了缓存的数据。

我一直在为这个问题寻找复制但没有成功,所以也许我错过了一些东西(我刚刚进入 java 世界)。

我的文件

pom.xml

<dependency>
    <groupId>org.hibernate</groupId>
    <artifactId>hibernate-validator</artifactId>
    <version>4.3.0.Final</version>
</dependency>
<dependency>
    <groupId>org.hibernate</groupId>
    <artifactId>hibernate-entitymanager</artifactId>
    <version>4.1.4.Final</version>
</dependency>
<dependency>
    <groupId>org.hibernate</groupId>
    <artifactId>hibernate-ehcache</artifactId>
    <version>4.1.4.Final</version>
</dependency>
<dependency>
    <groupId>net.sf.ehcache</groupId>
    <artifactId>ehcache-core</artifactId>
    <version>2.5.2</version>
</dependency>
<dependency>
    <groupId>org.terracotta</groupId>
    <artifactId>ehcache-probe</artifactId>
    <version>1.0.2</version>
</dependency>

实体类

package myPrototype.entities

import javax.persistence.Cacheable;
import javax.persistence.Entity;
import javax.persistence.NamedQueries;
import javax.persistence.NamedQuery;
import javax.persistence.Table;
import javax.xml.bind.annotation.XmlRootElement;
import org.hibernate.annotations.Cache;
import org.hibernate.annotations.CacheConcurrencyStrategy;

@Entity @Cacheable
@Cache(usage=CacheConcurrencyStrategy.READ_ONLY)
@Table(name = "Cars")
@XmlRootElement
@NamedQueries({
    @NamedQuery(name = "Cars.findAll", query = "SELECT d FROM Cars d"),
    [...]
public class Cars implements Serializable {
    private static final long serialVersionUID = 1L;
    @Basic(optional = false)
        @NotNull
        @Column(name = "id")
    @ReferenceField
    private int id;
    [...]

ehcache.xml

<?xml version="1.0" encoding="UTF-8"?>
<ehcache>
  <cache name="myPrototype.entities.Cars"
    maxElementsInMemory="500"
    eternal="false"
    overflowToDisk="false"
    timeToIdleSeconds="60"
    timeToLiveSeconds="120" />
</ehcache>

* 我尝试设置了永恒=“真”,但结果还是一样

persistence.xml

<?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="MyPrototype" transaction-type="JTA">
    <provider>org.hibernate.ejb.HibernatePersistence</provider>
    <jta-data-source>java:app/jdbc/mysqlserver/prototype</jta-data-source>
    <exclude-unlisted-classes>false</exclude-unlisted-classes>
    <shared-cache-mode>ENABLE_SELECTIVE</shared-cache-mode>
    <properties>
      <!-- property name="hibernate.cache.region.factory_class" value="org.hibernate.cache.ehcache.SingletonEhCacheRegionFactory"/ -->
      <property name="hibernate.cache.region.factory_class" value="org.hibernate.cache.ehcache.EhCacheRegionFactory" />
      <property name="hibernate.cache.use_second_level_cache" value="true"/>
      <property name="hibernate.cache.use_query_cache" value="true"/>
      <property name="hibernate.cache.region_prefix" value=""/>
    </properties>
  </persistence-unit>
</persistence>

* 我尝试在 SingletonEhCacheRegionFactory 和 EhCacheRegionFactory 之间切换

EJB

import myPrototype.entities.Cars
import java.util.Date;
import java.util.List;
import javax.ejb.Stateless;
import javax.persistence.EntityManager;
import javax.persistence.PersistenceContext;

@Stateless
public class CarsEJB {

    @PersistenceContext(unitName="MyPrototype")
    private EntityManager em;

    @Override
    public List<Cars> getCars() {
        return (List<Cars>)em.createNamedQuery("Cars.findAll").setMaxResults(200).getResultList();
    }

}

* 添加以阐明我如何运行查询

就像我说的,我可以通过 EHCache 监视器看到内存中的项目计数,所以我注意到以下几点: - ehcache 配置是属性加载的元素即将到期 - 当查询运行时,元素过期不会被重置(它们在第一次查询后 N 秒过期)。

再次;我是 java 的新手,所以假设我可能缺少基本概念。

感谢阅读!

【问题讨论】:

  • 你正在执行什么代码,你期望它做什么,它会做什么?
  • @JBNizet 例如;当我不止一次运行查询(通过工作中的 Restful 服务)并获取对象时,它会从数据库中检索数据而忽略缓存的对象。
  • 我认为您还应该通过在persistance.xml中设置属性进行调查 并检查休眠统计信息,这将帮助您检查查询原因没有命中缓存

标签: java hibernate jakarta-ee jpa ehcache


【解决方案1】:

Hibernate 永远不会对您的缓存执行查询。它可以做的是在数据库上执行查询,缓存结果,并在下次执行相同查询时返回这些缓存结果,使用相同的参数。但要做到这一点,每次执行的查询都必须是可缓存的。

使用 JPA API,可以使用查询提示、直接在指定的查询定义上或每次创建查询时完成:

@NamedQuery(name = "Cars.findAll", 
            query = "SELECT d FROM Cars d",
            hints = {@QueryHint(name = "org.hibernate.cacheable", value = "true")})

现在查询是可缓存的,Hibernate 将在第一次执行此查询时从数据库中加载汽车的 ID,然后从查询缓存中加载。然后相应的汽车本身将从实体缓存中加载。

【讨论】:

  • 太棒了!这正是我想要的,它有效!非常感谢!。现在,有没有办法默认为所有查询设置提示?
  • 你真的不想这样。很多查询会查询到频繁变化的数据,缓存会不断失效。此外,查询返回的每个实体都必须在二级缓存中,否则您将遇到大 N + 1 问题。当然,您有不能缓存的查询,因为它们依赖于当前时间,或者采用不断变化的参数。数据库很快。仅在必要时进行优化。
  • 很公平!我正在考虑明确禁用提示,但你是对的,除了那里有提示更具可读性。再次感谢!
猜你喜欢
  • 2011-07-27
  • 2014-08-21
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2015-10-22
  • 1970-01-01
相关资源
最近更新 更多