我正在尝试将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>
* 我尝试设置eternal =“true”,但结果仍然相同
的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的新手,所以我可能会错过基本的概念。
感谢阅读!
答案 0 :(得分:7)
Hibernate永远不会对缓存执行查询。它可以做的是对数据库执行查询,缓存结果,并在下次执行相同查询时使用相同的参数返回这些缓存的结果。但要做到这一点,查询每次执行时都必须是可缓存的。
使用JPA API,可以使用查询提示,直接在命名查询定义上或每次创建查询时完成:
@NamedQuery(name = "Cars.findAll",
query = "SELECT d FROM Cars d",
hints = {@QueryHint(name = "org.hibernate.cacheable", value = "true")})
现在查询是可缓存的,Hibernate将在第一次执行此查询时从数据库加载汽车的ID,然后从查询缓存加载。然后将从实体缓存中加载相应的汽车本身。