上一篇我們學會了如何使用Spring Boot使用進程內緩存在加速數據訪問。可能大家會問,那我們在Spring Boot中到底使用了什麼緩存呢?

在Spring Boot中通過 @EnableCaching 註解自動化配置合適的緩存管理器(CacheManager),Spring Boot根據下面的順序去偵測緩存提供者:

  • Generic
  • JCache (JSR-107) (EhCache 3, Hazelcast, Infinispan, and others)
  • EhCache 2.x
  • Hazelcast
  • Infinispan
  • Couchbase
  • Redis
  • Caffeine
  • Simple

除了按順序偵測外,我們也可以通過配置屬性spring.cache.type來強制指定。我們也可以通過debug調試查看 cacheManager 對象的實例來判斷當前使用了什麼緩存。在上一篇中,我們也展示瞭如何去查看當前使用情況。

當我們不指定具體其他第三方實現的時候,Spring Boot的Cache模塊會使用 ConcurrentHashMap 來存儲。而實際生產使用的時候,因爲我們可能需要更多其他特性,往往就會採用其他緩存框架,所以接下來我們會分幾篇分別介紹幾個常用優秀緩存的整合與使用。

使用EhCache

本篇我們將介紹如何在Spring Boot中使用EhCache進程內緩存。這裏我們將沿用上一篇的案例結果來進行改造,以實現EhCache的使用。

先回顧下這個基礎案例的三個部分:

User實體的定義

@Entity
@Data
@NoArgsConstructor
public class User {

    @Id
    @GeneratedValue
    private Long id;

    private String name;
    private Integer age;

    public User(String name, Integer age) {
        this.name = name;
        this.age = age;
    }
}

User實體的數據訪問實現(涵蓋了緩存註解)

@CacheConfig(cacheNames = "users")
public interface UserRepository extends JpaRepository<User, Long> {

    @Cacheable
    User findByName(String name);

}

測試驗證用例(涵蓋了CacheManager的注入,可用來觀察使用的緩存管理類)

@Slf4j
@RunWith(SpringRunner.class)
@SpringBootTest
public class Chapter51ApplicationTests {

    @Autowired
    private UserRepository userRepository;

    @Autowired
    private CacheManager cacheManager;

    @Test
    public void test() throws Exception {
        // 創建1條記錄
        userRepository.save(new User("AAA", 10));

        User u1 = userRepository.findByName("AAA");
        System.out.println("第一次查詢:" + u1.getAge());

        User u2 = userRepository.findByName("AAA");
        System.out.println("第二次查詢:" + u2.getAge());
    }

}

接下來我們通過下面的幾步操作,就可以輕鬆的把上面的緩存應用改成使用ehcache緩存管理。

第一步:在 pom.xml 中引入ehcache依賴

<dependency>
    <groupId>net.sf.ehcache</groupId>
    <artifactId>ehcache</artifactId>
</dependency>

在Spring Boot的parent管理下,不需要指定具體版本,會自動採用Spring Boot中指定的版本號。

第二步:在 src/main/resources 目錄下創建: ehcache.xml

<ehcache xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:noNamespaceSchemaLocation="ehcache.xsd">

    <cache name="users"
           maxEntriesLocalHeap="200"
           timeToLiveSeconds="600">
    </cache>

</ehcache>

完成上面的配置之後,再通過debug模式運行單元測試,觀察此時CacheManager已經是EhCacheManager實例,說明EhCache開啓成功了。或者在測試用例中加一句CacheManager的輸出,比如:

@Autowired
private CacheManager cacheManager;

@Test
public void test() throws Exception {
    System.out.println("CacheManager type : " + cacheManager.getClass());

    userRepository.save(new User("AAA", 10));

    User u1 = userRepository.findByName("AAA");
    System.out.println("第一次查詢:" + u1.getAge());

    User u2 = userRepository.findByName("AAA");
    System.out.println("第二次查詢:" + u2.getAge());
}

執行測試輸出可以得到:

CacheManager type : class org.springframework.cache.ehcache.EhCacheCacheManager
Hibernate: select next_val as id_val from hibernate_sequence for update
Hibernate: update hibernate_sequence set next_val= ? where next_val=?
Hibernate: insert into user (age, name, id) values (?, ?, ?)
2020-07-14 18:09:28.465  INFO 58538 --- [           main] o.h.h.i.QueryTranslatorFactoryInitiator  : HHH000397: Using ASTQueryTranslatorFactory
Hibernate: select user0_.id as id1_0_, user0_.age as age2_0_, user0_.name as name3_0_ from user user0_ where user0_.name=?
第一次查詢:10
第二次查詢:10

可以看到:

  1. 第一行輸出的CacheManager type爲 org.springframework.cache.ehcache.EhCacheCacheManager ,而不是上一篇中的 ConcurrentHashMap 了。
  2. 第二次查詢的時候,沒有輸出SQL語句,所以是走的緩存獲取

整合成功!

代碼示例

本文的相關例子可以查看下面倉庫中的 chapter5-2 目錄:

如果您覺得本文不錯,歡迎 Star 支持,您的關注是我堅持的動力!

相關文章