Java JPA&x2B;冬眠+;EHCache,意外行为
我试图在我的工作原型中实现EHCache,其中我有一个javax.persistence.Entity,表示我的数据库(MySQL,MySQL-connector-java-5.1.20.jar)上的一个表,它可以作为XML提供给消费者 我面临的问题是,Hibernate似乎仍在从数据库检索数据,即使EHCache将查询结果存储在内存中 我使用查看内存中的项目计数,并在缓存过期之前直接更改数据库中的数据,以了解缓存的数据是否实际被使用 我一直在寻找针对这个问题的复制,但没有成功,所以可能我错过了一些东西(我刚刚进入java世界) 我的档案 pom.xmlJava JPA&x2B;冬眠+;EHCache,意外行为,java,hibernate,jakarta-ee,jpa,ehcache,Java,Hibernate,Jakarta Ee,Jpa,Ehcache,我试图在我的工作原型中实现EHCache,其中我有一个javax.persistence.Entity,表示我的数据库(MySQL,MySQL-connector-java-5.1.20.jar)上的一个表,它可以作为XML提供给消费者 我面临的问题是,Hibernate似乎仍在从数据库检索数据,即使EHCache将查询结果存储在内存中 我使用查看内存中的项目计数,并在缓存过期之前直接更改数据库中的数据,以了解缓存的数据是否实际被使用 我一直在寻找针对这个问题的复制,但没有成功,所以可能我错过了
<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>
<?xml version="1.0" encoding="UTF-8"?>
<ehcache>
<cache name="myPrototype.entities.Cars"
maxElementsInMemory="500"
eternal="false"
overflowToDisk="false"
timeToIdleSeconds="60"
timeToLiveSeconds="120" />
</ehcache>
<?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>
ehcache.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>
<?xml version="1.0" encoding="UTF-8"?>
<ehcache>
<cache name="myPrototype.entities.Cars"
maxElementsInMemory="500"
eternal="false"
overflowToDisk="false"
timeToIdleSeconds="60"
timeToLiveSeconds="120" />
</ehcache>
<?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>
*我试着设置external=“true”,但结果还是一样
persistence.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>
<?xml version="1.0" encoding="UTF-8"?>
<ehcache>
<cache name="myPrototype.entities.Cars"
maxElementsInMemory="500"
eternal="false"
overflowToDisk="false"
timeToIdleSeconds="60"
timeToLiveSeconds="120" />
</ehcache>
<?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>
org.hibernate.ejb.HibernatePersistence
java:app/jdbc/mysqlserver/prototype
假的
启用\u选择性
*我尝试在SingletoneCacheRegionFactory和eCacheRegionFactory之间切换
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();
}
}
导入myPrototype.entities.Cars
导入java.util.Date;
导入java.util.List;
导入javax.ejb.Stateless;
导入javax.persistence.EntityManager;
导入javax.persistence.PersistenceContext;
@无国籍
公共类汽车{
@PersistenceContext(unitName=“MyPrototype”)
私人实体管理者;
@凌驾
公共列表getCars(){
return(List)em.createNamedQuery(“Cars.findAll”).setMaxResults(200.getResultList();
}
}
*添加以澄清如何运行查询
正如我所说,我可以通过EHCache monitor查看内存中的项目计数,因此我注意到以下几点:
-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,然后从查询缓存加载汽车ID。然后将从实体缓存中加载相应的汽车本身。您正在执行什么代码,您希望它做什么,它会做什么;当我多次运行查询(通过运行的Restful服务)并获取对象时,它总是从数据库中检索数据,第一次忽略缓存的对象。我认为您还应该通过在persistence.xml中设置属性并检查hibernate统计信息来进行调查,这将帮助您检查为什么查询不是很好!这正是我想要的,而且很有效!非常感谢!。现在,有没有办法在默认情况下将提示设置为所有查询?您真的不希望这样。许多查询将查询频繁更改的数据,缓存将不断失效。此外,查询返回的每个实体都必须在二级缓存中,否则就会出现一个大的N+1问题。当然,有些查询不能缓存,因为它们依赖于当前时间,或者采用不断变化的参数。数据库速度很快。仅在必要时进行优化。足够公平!我在考虑显式禁用提示,但你是对的,而且提示在那里更可读。再次感谢!