前言

Hi, everybody!

no time long see!!

相信大家在日常开发中一定多多少少会遇到一些问题,比如本地开发,后端服务响应很快!!

可是一放到线上我们访问呢就会很慢,这时我们就会对我们的后端服务进行优化,比如增加索引使查询更快 使某个命中的机率更高 !

但是则这样我们同样要去访问数据库,这就没达到我们优化的要求。

优化:

1,增加索引

2,尽量减少数据库的查库操作

所以这时我们就会引入缓存(cache)这个概念

什么是缓存?

缓存就是数据交换的缓冲区(称作Cache),是存贮数据(使用频繁的数据)的临时地方。

当用户查询数据,首先在缓存中寻找,如果找到了则直接返回。如果找不到,则去数据库中查找。

什么是SpringCache?

由Spring提供的利用AOP实现基于注解的缓存的框架。

它进行了合理的抽象,只需要一个注解我们就能轻松实现缓存,没有繁琐的配置文件

Spring Cache支持多种缓存实现,包括内存、Redis、Memcached等。开发人员可以根据自己的需求选择合适的缓存实现。

为什么要用缓存?

  1. 提高应用程序的性能:当应用程序需要访问数据库或其他外部资源时,如果这些资源的数据经常变化,那么每次请求都需要重新获取数据,这会导致应用程序的响应时间变慢。使用缓存可以将常用的数据存储在高速缓存中,这样下一次请求相同的数据时就可以直接从缓存中获取,而不需要再次查询数据库或其他外部资源,从而提高了应用程序的性能。
  2. 减少数据库负载:当应用程序频繁地访问数据库时,会给数据库带来很大的压力,导致数据库响应变慢或者崩溃。使用缓存可以将一些常用的数据存储在高速缓存中,这样下一次请求相同的数据时就可以直接从缓存中获取,而不需要再次查询数据库,从而减少了对数据库的负载。
  3. 提高用户体验:当应用程序的响应速度变快时,用户的体验也会得到提升。使用缓存可以避免用户等待过长时间才能看到页面的变化,从而提高了用户的满意度。
  4. 支持高并发访问:当多个用户同时访问应用程序时,如果每个用户都需要访问数据库或其他外部资源,那么会对系统造成很大的负担。使用缓存可以将一些常用的数据存储在高速缓存中,这样多个用户同时访问时就可以共享缓存中的数据,从而减轻了系统的负担。

使用配置

常用注解

  • @EnableCaching:开启缓存功能,一般放在启动类上。
  • @CacheConfig:当我们需要缓存的地方越来越多,你可以使用@CacheConfig(cacheNames = {"cacheName"})注解在 class 之上来统一指定value的值
  • @Cacheable:注解表示这个方法有了缓存的功能,方法的返回值会被缓存下来,下一次调用该方法前,会去检查是否缓存中已经有值,如果有就直接返回,不调用方法。如果没有,就调用方法,然后把结果缓存起来。这个注解一般用在查询方法上。

属性/方法名

解释

value

缓存名,必填,它指定了你的缓存存放在哪块命名空间

cacheNames

与 value 差不多,二选一即可

key

可选属性,可以使用 SpEL 标签自定义缓存的key

keyGenerator

key的生成器。key/keyGenerator二选一使用

cacheManager

指定缓存管理器

cacheResolver

指定获取解析器

condition

条件符合则缓存

unless

条件符合则不缓存

sync

是否使用异步模式,默认为false

  • @CachePut:使用该注解标志的方法,每次都会执行,并将结果存入指定的缓存中。其他方法可以直接从响应的缓存中读取缓存数据,而不需要再去查询数据库。一般用在新增方法上。

属性/方法名

解释

value

缓存名,必填,它指定了你的缓存存放在哪块命名空间

cacheNames

与 value 差不多,二选一即可

key

可选属性,可以使用 SpEL 标签自定义缓存的key

keyGenerator

key的生成器。key/keyGenerator二选一使用

cacheManager

指定缓存管理器

cacheResolver

指定获取解析器

condition

条件符合则缓存

unless

条件符合则不缓存

  • @CacheEvict:使用该注解标志的方法,会清空指定的缓存。一般用在更新或者删除方法上

属性/方法名

解释

value

缓存名,必填,它指定了你的缓存存放在哪块命名空间

cacheNames

与 value 差不多,二选一即可

key

可选属性,可以使用 SpEL 标签自定义缓存的key

keyGenerator

key的生成器。key/keyGenerator二选一使用

cacheManager

指定缓存管理器

cacheResolver

指定获取解析器

condition

条件符合则缓存

allEntries

是否清空所有缓存,默认为 false。如果指定为 true,则方法调用后将立即清空所有的缓存

beforeInvocation

是否在方法执行前就清空,默认为 false。如果指定为 true,则在方法执行前就会清空缓存

  • .@Caching:该注解可以实现同一个方法上同时使用多种注解。

代码实例

1,引入springcache的maven依赖

<dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-cache</artifactId>
        </dependency>

 2,在启动类使用注解缓存功能

package com.example.cache;

import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.cache.annotation.EnableCaching;

@EnableCaching
@SpringBootApplication
public class SpringBootCacheApplication {

    public static void main(String[] args) {
        SpringApplication.run(SpringBootCacheApplication.class, args);
    }

}

 3,配置我们的crud service接口和impl

package com.example.cache.service;

import com.example.cache.entity.User;
import com.github.pagehelper.PageInfo;

import java.util.List;

/**
 * @Program: SpringBoot
 * @ClassName UserService
 * @Author: liutao
 * @Description: user 逻辑接口
 * @Create: 2023-11-16 00:03
 * @Version 1.0
 **/


public interface UserService {
    User save(User user);

    PageInfo<User> page(Integer page,Integer pageSize);

    User findById(Integer id);

    List<User> findAll();

    int delete(Integer id);
}
package com.example.cache.service.impl;

import com.example.cache.entity.User;
import com.example.cache.mapper.UserMapper;
import com.example.cache.service.UserService;
import com.github.pagehelper.PageHelper;
import com.github.pagehelper.PageInfo;
import lombok.extern.slf4j.Slf4j;
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.annotation.Caching;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional;

import java.util.List;

/**
 * @Program: SpringBoot
 * @ClassName UserServiceImpl
 * @Author: liutao
 * @Description: user 实现类
 * @Create: 2023-11-16 00:06
 * @Version 1.0
 **/
@Slf4j
@Service
public class UserServiceImpl implements UserService {
    @Autowired
    private UserMapper userMapper;

    @Transactional(rollbackFor = {Exception.class})
    @Caching(put = {@CachePut(value = "userCache", key = "#user.id", unless = "#result == null")}, evict = {@CacheEvict(value = "allUsersCache", allEntries = true)})
    @Override
    public User save(User user) {
        int num = 0;
        if (user.getId() == null) {
            num = userMapper.insert(user);
        } else {
            num = userMapper.update(user);
        }
        log.info("User " + user.getId());
        return num == 1 ? userMapper.findById(user.getId()) : null;
    }

    @Cacheable(value = "allUsersCache", unless = "#result == null")
    @Override
    public PageInfo<User> page(Integer page, Integer pageSize) {
        // 开启分页
        PageHelper.startPage(page, pageSize);
        List<User> users = userMapper.findAll();
        if (users != null && !users.isEmpty()) {
            return new PageInfo<>(users);
        }
        return null;
    }

    @Cacheable(value = "userCache", key = "#id", unless = "#result == null")
    @Override
    public User findById(Integer id) {
        return userMapper.findById(id);
    }

    @Cacheable(value = "allUsersCache", unless = "#result == null")
    @Override
    public List<User> findAll() {
        return userMapper.findAll();
    }

    @Caching(evict = {@CacheEvict(value = "userCache", key = "#id"), @CacheEvict(value = "allUsersCache", allEntries = true),})
    @Override
    public int delete(Integer id) {
        return userMapper.delete(id);
    }
}

 4,web层

package com.example.cache.controller;

import com.example.cache.entity.User;
import com.example.cache.service.UserService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.*;

import java.util.List;

/**
 * @Program: SpringBoot
 * @ClassName UserController
 * @Author: liutao
 * @Description: user api
 * @Create: 2023-11-16 00:00
 * @Version 1.0
 **/
@RequestMapping("/user")
@RestController
public class UserController {
    @Autowired
    private UserService userService;

    @GetMapping
    List<User> findAll() {
        return userService.findAll();
    }

    @GetMapping("/page")
    Object page(Integer page,Integer pageSize) {
        return userService.page(page,pageSize);
    }

    @GetMapping("/{id}")
    User findById(@PathVariable Integer id) {
        return userService.findById(id);
    }

    @PostMapping("/save")
    User save(User user) {
        return userService.save(user);
    }


    @DeleteMapping("/{id}")
    int delete(@PathVariable Integer id) {
        return userService.delete(id);
    }

}

 效果图

1,打开服务:访问两次 http://localhost:8080/user

springboot启动Launcher枚举缓存 springboot开启缓存_User

springboot启动Launcher枚举缓存 springboot开启缓存_User_02

2,清空日志再次访问,我们可以看到这次是从缓存查出来的,并没有进行查库操作

springboot启动Launcher枚举缓存 springboot开启缓存_缓存_03