如何使用JPA2的@Cacheable代替Hibernate的@Cache

时间:2010-09-08 01:20:43

标签: hibernate orm jpa caching jpa-2.0

通常,我使用Hibernate的 @Cache(usage = CacheConcurrencyStrategy.NONSTRICT_READ_WRITE)来缓存@Entity类,它运行良好。

在JPA2中,还有另一个@Cacheable注释似乎与Hibernate的@Cache具有相同的功能。为了使我的实体类独立于hibernate的包,我想尝试一下。但我不能让它发挥作用。每次简单的id查询仍然会访问数据库。

有谁能告诉我哪里出错了?感谢。

实体类:

@Entity
//@Cache(usage = CacheConcurrencyStrategy.NONSTRICT_READ_WRITE)
@Cacheable(true) 
public class User implements Serializable
{
 // properties
}

测试类:

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(locations={"classpath:app.xml"})
@TransactionConfiguration(transactionManager="transactionManager")
public class UserCacheTest
{
  @Inject protected UserDao userDao;

  @Transactional
  @Test
  public void testGet1()
  {
    assertNotNull(userDao.get(2L));
  }

  @Transactional
  @Test
  public void testGet2()
  {
    assertNotNull(userDao.get(2L));
  }

  @Transactional
  @Test
  public void testGet3()
  {
    assertNotNull(userDao.get(2L));
  }
}

测试结果显示每个“get”命中数据库层(hibernate.show_sql = true)。

Persistence.xml:

<property name="hibernate.dialect" value="org.hibernate.dialect.MySQLDialect"/>
<property name="hibernate.show_sql" value="true"/>
<property name="hibernate.format_sql" value="true" />
<property name="hibernate.use_outer_join" value="true"/>

<property name="hibernate.cache.provider_class" value="org.hibernate.cache.SingletonEhCacheProvider"/>
<property name="hibernate.cache.use_second_level_cache" value="true"/>
<property name="hibernate.cache.use_query_cache" value="true"/>

JPA代码:

@Override
public T get(Serializable id)
{
  return em.find(clazz, id);
}

2 个答案:

答案 0 :(得分:44)

根据JPA 2.0规范,如果要使用@Cacheable注释有选择地缓存实体,则应在<shared-cache-mode>(或等效{persistence.xml中指定javax.persistence.sharedCache.mode创建EntityManagerFactory时{1}}。

下面是具有相关元素和属性的示例persistence.xml

<persistence 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" version="2.0">
  <persistence-unit name="FooPu" transaction-type="RESOURCE_LOCAL">
    <provider>org.hibernate.ejb.HibernatePersistence</provider>
    ...
    <shared-cache-mode>ENABLE_SELECTIVE</shared-cache-mode>
    <properties>
      ...
      <property name="hibernate.cache.provider_class" value="org.hibernate.cache.SingletonEhCacheProvider"/>
      <property name="hibernate.cache.use_second_level_cache" value="true"/>
      <property name="hibernate.cache.use_query_cache" value="true"/>
    </properties>
  </persistence-unit>
</persistence>

请注意,我至少看到了一个与缓存相关的问题HHH-5303。所以不保证以上内容:)

参考

  • Hibernate EntityManager参考指南
  • JPA 2.0规范
    • 第3.7.1节“共享缓存模式元素”
    • 第11.1.7节“可缓存注释”

答案 1 :(得分:11)

对于使用Spring配置而不是persistence.xml的用户,这里有一个示例:

<bean id="entityManagerFactory" class="org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean">
    <property name="dataSource" ref="dataSource"/>
    <property name="jpaVendorAdapter">
        <bean class="org.springframework.orm.jpa.vendor.HibernateJpaVendorAdapter">
            <property name="database" value="MYSQL"/>
            <property name="databasePlatform" value="org.hibernate.dialect.MySQL5InnoDBDialect"/>
            <property name="showSql" value="true"/>
            <property name="generateDdl" value="false"/>
        </bean>
    </property>
    <property name="packagesToScan" value="com.mycompany.myproject.domain"/>
    <property name="jpaPropertyMap">
        <map>
            <entry key="hibernate.cache.region.factory_class" value="org.hibernate.cache.ehcache.SingletonEhCacheRegionFactory"/>
            <entry key="hibernate.cache.use_second_level_cache" value="true"/>
            <entry key="hibernate.cache.use_query_cache" value="true"/>
            <entry key="javax.persistence.sharedCache.mode" value="ENABLE_SELECTIVE" />
        </map>
    </property>
</bean>

另请注意,如果您使用@Cacheable注释,则只能使用默认缓存并发策略,该策略由getDefaultAccessType()的{​​{1}}方法确定。在EhCache的情况下,它是READ_WRITE。如果您想使用其他策略,则必须使用Hibernate的RegionFactory注释。