Spring boot Caffeine缓存

本文介绍了如何在Spring Boot应用中集成并配置Caffeine缓存,包括引入依赖、启用缓存注解和配置,以及讲解了@Cacheable、@CacheEvict和@CachePut三个关键注解的用法,最后展示了如何通过API直接操作缓存。

1. 引入jar:

	implementation 'com.github.ben-manes.caffeine:caffeine:2.8.4'
	implementation('org.springframework.boot:spring-boot-starter-cache')

2. 使用配置类配置

2.1启动类加注解: @EnableCaching

2.2 配置类:

package com.cnpc.cmp.cmpworksheetapp.config;

import com.github.benmanes.caffeine.cache.*;
import org.checkerframework.checker.nullness.qual.NonNull;
import org.checkerframework.checker.nullness.qual.Nullable;
import org.springframework.cache.CacheManager;
import org.springframework.cache.caffeine.CaffeineCacheManager;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

import java.util.Collections;
import java.util.concurrent.TimeUnit;

/**
 * @ClassName CacheConfig
 * @Description springboot集成本地缓存caffeine 配置类 
 * @Date
 */

@Configuration
public class CacheConfig {

    @Bean("caffeineCacheManager")
    public CacheManager cacheManager(){
        CaffeineCacheManager caffeineCacheManager = new CaffeineCacheManager();
        caffeineCacheManager.setCaffeine(getCaffeine());
        caffeineCacheManager.setCacheLoader(cacheLoader());
        //是否允许值为空
        caffeineCacheManager.setAllowNullValues(false);
        caffeineCacheManager.setCacheNames(Collections.singleton("test"));
        return caffeineCacheManager;
    }



   @Bean
   public CacheLoader<Object,Object> cacheLoader(){
        return new CacheLoader<Object, Object>() {
            @Override
            public @Nullable Object load(@NonNull Object key) throws Exception {
                // 这里我们就可以从数据库或者其他地方查询最新的数据  ???
                return null;
            }
            @Override
            public @Nullable Object reload(@NonNull Object key, @NonNull Object oldValue) throws Exception {
                System.out.println("--refresh--:"+key);
                return oldValue;
            }
        };

   }

    @Bean
    public LoadingCache getCache(){
        LoadingCache build = getCaffeine().build(cacheLoader());
        return build;
    }


   @Bean
   public Caffeine getCaffeine(){
       Caffeine caffeine =  Caffeine.newBuilder()
               //设置过期时间。最后一次写入或访问后过多久过期。
               .expireAfterAccess(60, TimeUnit.SECONDS)
               //cache的初始容量值
               .initialCapacity(100)
               //maximumSize用来控制cache的最大缓存数量,maximumSize和maximumWeight(最大权重)不可以同时使用,
               .maximumSize(1000)
               //弱引用
//                     .weakKeys()
//                     .weakValues()
               //创建或更新之后多久刷新,需要设置cacheLoader
               .refreshAfterWrite(10,TimeUnit.SECONDS)
               //缓存写入/删除 监控
               .writer(new CacheWriter<Object, Object>() {

                   @Override
                   public void write(@NonNull Object key, @NonNull Object value) {
                       System.out.println("缓存写入: key= "+key+";value="+value);
                   }

                   @Override
                   public void delete(@NonNull Object key, @Nullable Object value, @NonNull RemovalCause cause) {
                       System.out.println("缓存删除: key= "+key+";value="+value);

                   }
               });
        return caffeine;
   }

}

3. 使用注解使用缓存:

其中注解解释:

  • @Cacheable 触发缓存入口(这里一般放在创建和获取的方法上),当缓存中有是,方法不会执行,直接取缓存中的。如果没有,执行方法然后将返回值存入到缓存中。
  • @CacheEvict 触发缓存的eviction(用于删除的方法上)
  • @CachePut 更新缓存且不影响方法执行(用于修改的方法上,该注解下的方法始终会被执行)
import com.github.benmanes.caffeine.cache.Cache;
import com.github.benmanes.caffeine.cache.CacheLoader;
import com.github.benmanes.caffeine.cache.Caffeine;
import com.github.benmanes.caffeine.cache.LoadingCache;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.BeanUtils;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.cache.annotation.CacheEvict;
import org.springframework.cache.annotation.CachePut;
import org.springframework.cache.annotation.Cacheable;
import org.springframework.cache.caffeine.CaffeineCache;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional;
import java.util.ArrayList;
import java.util.Optional;

 
@Service
public class CacheServiceImpl implements CacheService {

    private Logger logger = LoggerFactory.getLogger(this.getClass());
    
    @Autowired
    private UserHolidayRepository userHolidayRepository;

    @Override
    public UserHoliday getByName(String name) {
        return null;
    }

    /**
     * 将数据存放到缓存中
     * 创建缓存:@Cacheable注解即可自动将数据添加到缓存中,后续直接从缓存中读取数据。
     * value:表示缓存的名称;key:表示缓存的key,可以为空。
     *
     * @param id
     * @return
     */
    @Cacheable(value = "test", key = "#id")
    @Override
    public UserHoliday getById(String id) {
        logger.info("getById parameter id : {}", id);
        Optional<UserHoliday> optional = userHolidayRepository.findById(id);
        UserHoliday userHoliday = new UserHoliday();
        if (optional.isPresent()) {
            userHoliday = optional.get();
        }
        logger.info("getById return userHoliday:{}", userHoliday);
        return userHoliday;
    }


    /**
     * 将满足条件的数据存放到缓存中。
     *
     * @param id
     * @return
     * @Cacheable 有一个参数叫做condition,该条件为true时则放到缓存到。该参数同样需使用SpEL表达式。
     */
    @Cacheable(value = "test", key = "#id", condition = "#id.equals('2')")
    @Override
    public UserHoliday getByConditionalId(String id) {
        logger.info("getByConditionalId parameter id : {}", id);
        Optional<UserHoliday> optional = userHolidayRepository.findById(id);
        UserHoliday userHoliday = new UserHoliday();
        if (optional.isPresent()) {
            userHoliday = optional.get();
        }
        logger.info("getByConditionalId return userHoliday:{}", userHoliday);
        return userHoliday;
    }


    /**
     * 更新缓存
     *
     * @param holidayDTO
     * @return
     */
    @CachePut(value = "test",   key = "#id")
    @Override
    public UserHoliday updateRemainDay(String id,HolidayDTO holidayDTO) {
        logger.info("updateRemainDay parameter holidayDTO : {}", holidayDTO.toString());
      
        Optional<UserHoliday> byId = userHolidayRepository.findById(holidayDTO.getId());
        UserHoliday userHoliday = byId.get();
        BeanUtils.copyProperties(holidayDTO, userHoliday);
        UserHoliday returnUserHoliday = userHolidayRepository.saveAndFlush(userHoliday);
        logger.info("updateRemainDay return returnUserHoliday : {}", returnUserHoliday.toString());
        return returnUserHoliday;
    }

    @CacheEvict(value = "test",   key = "#id")
    @Override
    public void deleteById(String id) {
        userHolidayRepository.deleteById(id);

    }


}

4. 使用api操作缓存:

4.1注入:   @Autowired
    LoadingCache syncCache;

4.2 使用:

syncCache.put("a","aaa");
syncCache.getIfPresent("a");
ArrayList<String> strings = new ArrayList<>();
strings.add("a");
strings.add("b");
syncCache.getAllPresent(strings);


例如:
    @Autowired
    LoadingCache syncCache;

    

    /**
     * 将数据存放到缓存中
     * 创建缓存:@Cacheable注解即可自动将数据添加到缓存中,后续直接从缓存中读取数据。
     * value:表示缓存的名称;key:表示缓存的key,可以为空。
     
     */
    @Cacheable(value = "test", key = "#id")
    @Override
    public UserHoliday getById(String id) {

        syncCache.put("a","aaa");
        syncCache.put("b","dddd");
        logger.info("getById parameter id : {}", id);
        Optional<UserHoliday> optional = userHolidayRepository.findById(id);
        UserHoliday userHoliday = new UserHoliday();
        if (optional.isPresent()) {
            userHoliday = optional.get();
        }
        logger.info("getById return userHoliday:{}", userHoliday);
        return userHoliday;
    }


    /**
     * 将满足条件的数据存放到缓存中。
     *
     * @param id
     * @return
     * @Cacheable 有一个参数叫做condition,该条件为true时则放到缓存到。该参数同样需使用SpEL表达式。
     */
    @Cacheable(value = "test", key = "#id", condition = "#id.equals('2')")
    @Override
    public UserHoliday getByConditionalId(String id) {

        logger.info(" syncCache.getIfPresent  : {}", syncCache.getIfPresent("a"));
        ArrayList<String> strings = new ArrayList<>();
        strings.add("a");
        strings.add("b");
        logger.info(" syncCache.getAllPresent  : {}", syncCache.getAllPresent(strings));


        logger.info("getByConditionalId parameter id : {}", id);
        Optional<UserHoliday> optional = userHolidayRepository.findById(id);
        UserHoliday userHoliday = new UserHoliday();
        if (optional.isPresent()) {
            userHoliday = optional.get();
        }
        logger.info("getByConditionalId return userHoliday:{}", userHoliday);
        return userHoliday;
    }

 

 

 

 

 

 

Spring Boot Caffeine 是一个用于集成 Spring BootCaffeine 缓存库的插件。Caffeine 是一个高性能的 Java 缓存库,它提供了内存缓存的功能,可以减少对数据库或其他外部资源的访问次数,从而提高应用程序的性能。 要在 Spring Boot 中使用 Caffeine,首先需要添加以下 Maven 依赖项: ```xml <dependency> <groupId>com.github.ben-manes.caffeine</groupId> <artifactId>caffeine</artifactId> </dependency> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-cache</artifactId> </dependency> ``` 接下来,在你的 Spring Boot 应用程序的配置类上添加 `@EnableCaching` 注解,以启用缓存功能。然后,在需要缓存的方法上添加 `@Cacheable` 注解,指定缓存的名称和缓存的键。 例如,假设你有一个 UserService 类,其中有一个需要进行缓存的方法 getUserById: ```java @Service public class UserService { @Cacheable("users") public User getUserById(Long id) { // 从数据库或其他外部资源获取用户信息 return userRepository.findById(id); } } ``` 在上面的例子中,使用了 `@Cacheable("users")` 注解来指定缓存的名称为 "users",并且根据传入的 id 参数进行缓存。当该方法被调用时,如果缓存中已经存在相应的键值对,则直接从缓存中获取结果,否则会执行方法体内的代码,并将结果存入缓存中。 另外,你还可以使用 `@CacheEvict` 注解来清除缓存中的数据,例如: ```java @CacheEvict("users") public void deleteUserById(Long id) { // 删除用户操作 } ``` 上述代码中的 `@CacheEvict("users")` 注解会在 deleteUserById 方法执行后清除名为 "users" 的缓存。 通过以上的配置和注解,你就可以在 Spring Boot 应用程序中使用 Caffeine 缓存来提高性能了。当然,还有其他更多的配置选项和注解可以使用,你可以根据自己的需求进行调整和扩展。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值