Как использовать @Cacheable в JPA2 вместо @Cache в Hibernate - PullRequest
46 голосов
/ 08 сентября 2010

Обычно я использую @ Cache Hibernate (using = CacheConcurrencyStrategy.NONSTRICT_READ_WRITE) для кэширования класса @Entity, и это хорошо работает.

В JPA2 есть еще одна аннотация @Cacheable, которая похожа на ту же функциональность, что и @Cache Hibernate.Чтобы сделать мой класс сущности независимым от пакета hibernate, я хочу попробовать.Но я не могу заставить это работать.Каждый раз, когда простой запрос все еще попадает в БД.

Кто-нибудь может сказать мне, где что-то не так?Спасибо.

Класс сущности:

@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 ]

44 голосов
/ 08 сентября 2010

Согласно спецификации JPA 2.0, если вы хотите выборочно кэшировать объекты с помощью аннотации @Cacheable, вы должны указать <shared-cache-mode> в persistence.xml (или эквивалент javax.persistence.sharedCache.mode при создании EntityManagerFactory).

Ниже приведен образец 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 «Кэшируемая аннотация»
11 голосов
/ 27 марта 2014

Для тех, кто использует конфигурацию 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() из RegionFactory.В случае EhCache это READ_WRITE .Если вы хотите использовать другую стратегию, вы должны использовать аннотации @Cache Hibernate.

...