一级缓存不适用于 JPA 和 Hibernate

Ash*_*vin 3 java spring caching hibernate jpa

我是使用休眠缓存的新手(第一级、第二级和查询缓存)。

我的项目是使用 Spring MVC 和 JPA 配置的。

我正在使用下面的 JUnit 测试用例测试一级缓存。

public class FirstLevelCacheTest
{
    private static final Logger logger = LoggerFactory.getLogger(FirstLevelCacheTest.class);

    @PersistenceContext
    private EntityManager       entityManager;

    @Test
    public void testFirstLevelCacheTest()
    {
        Long clientId = 1L;

        // fetch the client entity from database first time
        Client client1 = entityManager.find(Client.class, clientId);
        logger.debug("Client Name : {}", client1.getName());

        // fetch the client entity again
        client1 = entityManager.find(Client.class, clientId);
        logger.debug("Client Name : {}", client1.getName());
    }
}
Run Code Online (Sandbox Code Playgroud)

我的实体类定义为:

@Entity
@JsonSerialize(include = JsonSerialize.Inclusion.NON_NULL)
@Table(name = "client")
public class Client extends BaseModel
{
    // instance variable declaration

    // getter and setter methods for instance variables
}
Run Code Online (Sandbox Code Playgroud)

如果默认启用一级缓存,这应该执行一次本机查询。但是我在执行此查询时得到以下结果:

Hibernate: select client0_.id as id0_0_, client0_.created as created0_0_, client0_.createdBy as createdBy0_0_, client0_.updated as updated0_0_, client0_.updatedBy as updatedBy0_0_, client0_.contactNo as contactNo0_0_, client0_.contactPerson as contactP7_0_0_, client0_.contactPersonEmail as contactP8_0_0_, client0_.contactPersonNo as contactP9_0_0_, client0_.email as email0_0_, client0_.name as name0_0_ from client client0_ where client0_.id=?
[main] DEBUG Client Name : Client1
Hibernate: select client0_.id as id0_0_, client0_.created as created0_0_, client0_.createdBy as createdBy0_0_, client0_.updated as updated0_0_, client0_.updatedBy as updatedBy0_0_, client0_.contactNo as contactNo0_0_, client0_.contactPerson as contactP7_0_0_, client0_.contactPersonEmail as contactP8_0_0_, client0_.contactPersonNo as contactP9_0_0_, client0_.email as email0_0_, client0_.name as name0_0_ from client client0_ where client0_.id=?
[main] DEBUG Client Name : Client1
Run Code Online (Sandbox Code Playgroud)

以下是我的持久化相关配置:

@Configuration
@EnableTransactionManagement
public class PersistanceConfig
{
    // injection

    private String[] PACKAGES_TO_SCAN = new String[] { "com.mypackage1", "com.mypackage2" };

    @Bean
    public DataSource dataSource()
    {
        // dataSource setting 'com.mysql.jdbc.Driver' as a jdbc driver
    }

    private Properties jpaProperties()
    {
        Properties properties = new Properties();

        properties.put(AvailableSettings.DIALECT, hibernateDialect);
        properties.put(AvailableSettings.SHOW_SQL, hibernateShowSql);

        // 2nd level cache
        properties.put("hibernate.cache.use_second_level_cache", true);
        properties.put("hibernate.cache.region.factory_class", "org.hibernate.cache.ehcache.EhCacheRegionFactory");

        return properties;
    }

    @Bean
    public LocalContainerEntityManagerFactoryBean entityManagerFactory()
    {
        // entity manager settings using dataSource and jpaProperties
    }

    @Bean
    public JpaTransactionManager transactionManager()
    {
        // transaction manager settings using entityManagerFactory
    }
}
Run Code Online (Sandbox Code Playgroud)

任何人都可以帮助我解决问题或我做错了什么吗?

提前致谢 !

Vla*_*cea 9

您需要@Transactional使用 Spring注释测试方法或使用 Spring TransactionTemplate

即使检索实体不需要事务,在一个内部对多个调用进行分组也会清楚地指定持久性上下文边界(它打算从哪里开始,应该在哪里结束)。

要复制第一级缓存,您需要在整个测试方法中使用相同的工作单元,并且注释测试@Transactional将启动一个事务(绑定到当前执行的持久性上下文事务的 Spring 逻辑事务,该事务与实际的物理数据库事务相关联) .

当测试方法结束时,事务通常会回滚,因此不会将更改传播到任何后续测试,但在当前执行的测试方法中,您可以看到自己的更改,即 ACID 中的隔离属性。

  • 我通常让我的服务层设置 @Transactional 边界,而不是 DAO 层。这是因为服务最终可能会调用您希望在同一事务中执行的多个 DAO。这也解决了您的循环 getById 问题,因为在事务内部您获得相同的 Hibernate Session,因此第一级缓存将始终返回相同的对象实例,无论您调用 session.get() 或 entityManager.find() 多少次 (2认同)