JPA + Hibernate + EHCache,出乎意料的行为

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

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

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

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

我的文件

的pom.xml

 org.hibernate hibernate-validator 4.3.0.Final   org.hibernate hibernate-entitymanager 4.1.4.Final   org.hibernate hibernate-ehcache 4.1.4.Final   net.sf.ehcache ehcache-core 2.5.2   org.terracotta ehcache-probe 1.0.2  

实体类

 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中

     

* 我尝试设置eternal =“true”,但结果仍然相同

persistence.xml中

    org.hibernate.ejb.HibernatePersistence java:app/jdbc/mysqlserver/prototype false ENABLE_SELECTIVE          

* 我尝试在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 getCars() { return (List)em.createNamedQuery("Cars.findAll").setMaxResults(200).getResultList(); } } 

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

就像我说的,我可以通过EHCache监视器看到内存中的项目数,所以我注意到以下内容: – ehcache配置是属性加载的元素即将到期 – 当查询运行时,元素到期未被重置(他们仍然活着)在第一次查询后N秒过期。

再次; 我是java的新手,所以我可能会错过基本概念。

谢谢阅读!

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

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

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

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