Spring Boot | Spring Boot 默认 “缓存管理“ 、Spring Boot “缓存注解“ 介绍

目录:

    • 一、Spring Boot 默认 "缓存" 管理 :
      • 1.1 基础环境搭建
        • ① 准备数据
        • ② 创建项目
        • ③ 编写 "数据库表" 对应的 "实体类"
        • ④ 编写 "操作数据库" 的 Repository接口文件
        • ⑤ 编写 "业务操作列" Service文件
        • ⑥ 编写 "application.properties配置文件"
        • ⑦ 项目测试 ( 实际开发中的"问题突显",用 "缓存技术" 能解决这个问题 )
      • 1.2 Spring Boot "默认缓存体验"
        • (1) 使用 "@EnableCaching" 注解 开始 “缓存管理”
        • (2) 使用 "@Cacheable( )" 注解对 "数据操作方法" 进行 “缓存管理” ( 将该注解放在service类的“操作方法”上,让其对"查询结果" 进行 “缓存” )
        • (3) Spring Boot默认缓存测试
    • 二、Spring Boot "缓存注解" 介绍 :
      • 2.1 @EnableCaching 注解 ( 用在 "主程序启动类" 上,用于开启SpringBoot的 "缓存管理支持")
      • 2.2 @Cacheable( ) 注解 ( 作用于"类" 或 "方法",通常用在 "数据查询" 功能的 "方法" 上 , 将 "查询结果" 存进 "缓存空间" 中 )
      • 2.3 @CachePut 注解 ( 作用于"类" 或 "方法",通常用在 "数据更新" 功能的 "方法" 上 ,在 "更新数据库" 后 "更新缓存" )
      • 2.4 @CacheEvict 注解 ( 作用于"类" 或 "方法",通常用在 "数据删除" 方法上 ,在”删除数据库中 数据“ 后,进行删除 “缓存数据” )
      • 2.5 @Caching注解 ( 作用于"类" 或 "方法" )
      • 2.6 @CacheConfig注解 ( 作用于"类" , 为该类下 的 "缓存注解" 设置 “公共属性” )

作者简介 :一只大皮卡丘,计算机专业学生,正在努力学习、努力敲代码中! 让我们一起继续努力学习!

该文章参考学习教材为:
《Spring Boot企业级开发教程》 黑马程序员 / 编著
文章以课本知识点 + 代码为主线,结合自己看书学习过程中的理解和感悟 ,最终成就了该文章

文章用于本人学习使用 , 同时希望能帮助大家。
欢迎大家点赞👍 收藏⭐ 关注💖哦!!!

(侵权可联系我,进行删除,如果雷同,纯属巧合)


  • 缓存分布式系统中的重要组件,主要 解决数据厍数据高并发访问问题。在实际开发中,尤其是用户访问量较大网站,为了提高服务器访问性能减少数据库的压力提高用户体验,使用 缓存显得尤为重要Spring Boot缓存提供了良好的支持

一、Spring Boot 默认 “缓存” 管理 :

  • Spring框架支持透明地应用程序添加缓存并对缓存进行管理,其管理缓存核心是将缓存 应用于操作数据的方法中,从而 减少操作 数据的次数同时不会对程序本身造成任何干扰
  • Spring Boot继承了Spring框架缓存管理功能,通过使用 @EnableCaching注解开启基于注解缓存支持Spring Boot可以启动缓存管理自动化配置。

1.1 基础环境搭建

  • 使用缓存主要目的减少数据库数据访问压力提高用户体验,下面代码例子将结合数据库访问操作Spring Boot缓存管理讲行演示说明
① 准备数据

先创建了一个 数据库springbootdata,然后创建了两个表 t_articlet_comment ,并向表中插入数据。
其中评论表t_commenta_id 与文章表t_article主键id 相关联 ( t_article主键作为t_comment表外键)。

springbootdata.sql

② 创建项目

创建项目,引入相关依赖
在这里插入图片描述

③ 编写 “数据库表” 对应的 “实体类”
  • 编写t_comment表对应的 实体类Comment,并用 JPA相关注解配置映射关系 :

    Comment.java

    package com.myh.chapter_14.domain;
    
    import jakarta.persistence.*;
    
    //指定该实现类映射的数据库表
        @Entity(name = "t_commet") //设置ORM实体类, 并指定对应的表明
    public class Comment {
    
        //表示数据库表中主键对应的属性
        @Id
        @GeneratedValue(strategy= GenerationType.IDENTITY) //设置主键的生成策略 (主键自增)
        private Integer id;
    
        @Column(name = "content") //指定映射的表字段名
        private String content;
    
        @Column(name = "author")
        private String author;
    
        @Column(name = "a_id")
        private Integer aId;
    
        public Integer getId() {
            return id;
        }
    
        public void setId(Integer id) {
            this.id = id;
        }
    
        public String getContent() {
            return content;
        }
    
        public void setContent(String content) {
            this.content = content;
        }
    
        public String getAuthor() {
            return author;
        }
    
        public void setAuthor(String author) {
            this.author = author;
        }
    
        public Integer getaId() {
            return aId;
        }
    
        public void setaId(Integer aId) {
            this.aId = aId;
        }
    
        @Override
        public String toString() {
            return "Comment{" +
                    "id=" + id +
                    ", content='" + content + '\'' +
                    ", author='" + author + '\'' +
                    ", aId=" + aId +
                    '}';
        }
    }
    
④ 编写 “操作数据库” 的 Repository接口文件
  • 项目中创建 repository包,在该包创建一个用于操作数据库自定义Repository接口该接口 继承JpaRepository
    ( Repository接口 中为 操作数据库方法 )


    CommentRepository.java

    package com.myh.chapter_14.Repository;
    
    import com.myh.chapter_14.domain.Comment;
    import org.springframework.data.jpa.repository.JpaRepository;
    import org.springframework.data.jpa.repository.Query;
    
    //Repository接口中为操作数据库的方法
    /*
      继承了JpaRepository接口,其中有操作数据库的curd方法,也用方法关键字的形式来操作数据库,或者使用@Query注解的方式来操作数据库
     */
    public interface CommentRepository extends JpaRepository<Comment,Integer> {
    
        //根据评论id来修改评论作者author
        //通过updateComment()方法对应的@Query注解来操作数据库
        @Query("update t_commet c set c.author = ?1 where c.id = ?2") //通过该标签来操作数据库
        public int updateComment(String author, Integer id);
    }
    
⑤ 编写 “业务操作列” Service文件
  • 项目中创建 service包,在该包创建一个用于操作Comment相关业务操作Service实体类

    CommentController.java

    package com.myh.chapter_14.controller;
    
    import com.myh.chapter_14.domain.Comment;
    import com.myh.chapter_14.service.CommentService;
    import org.springframework.beans.factory.annotation.Autowired;
    import org.springframework.stereotype.Controller;
    import org.springframework.web.bind.annotation.GetMapping;
    import org.springframework.web.bind.annotation.PathVariable;
    import org.springframework.web.bind.annotation.ResponseBody;
    import org.springframework.web.bind.annotation.RestController;
    
    //@ResponseBody注解 : 将方法的返回值转换为"指定类型",存入响应体中,然后响应给“前端”
    @RestController // 该注解等于 @ResponseBody 注解 + @Controller注解
    public class CommentController {
    
        @Autowired
        private CommentService commentService;
    
        /**
         * findById()方法
         */
        @GetMapping("/get/{id}") //路径变量,用@PathVariable()注解来接受路径变量
        public Comment findById(@PathVariable("id") int comment_id) {
            Comment comment = commentService.findById(comment_id);
            return comment;
        }
    
        /**
         * updateComment()方法
         */
        @GetMapping("/update/{id}/{author}") //路径变量,用@PathVariable()注解来接受路径变量
        public Comment updateComment(@PathVariable("id") int comment_id,@PathVariable("author") String author) {
            Comment comment = commentService.findById(comment_id);
            comment.setAuthor("张三");
            Comment updateComment = commentService.updateComment(comment);
            return updateComment;
        }
    
        /**
         * deleteComment()方法
         */
        @GetMapping("/delete/{id}}") //路径变量,用@PathVariable()注解来接受路径变量
        public void deleteComment(@PathVariable("id") int comment_id) {
            commentService.deleteComment(comment_id);
        }
    
    }
    
⑥ 编写 “application.properties配置文件”
  • application.properties

    #配置数据库信息
    spring.datasource.driver-class-name=com.mysql.cj.jdbc.Driver
    spring.datasource.url=jdbc:mysql://localhost:3306/springbootdata?useUnicode=true&characterEncoding=utf-8&serverTimezone=GMT&nullCatalogMeansCurrent=true
    spring.datasource.username=root
    spring.datasource.password=root
    
    #显示使用JPA进行"数据库查询"的SQL语句,用于展示操作的Sql语句
    #这个属性决定了是否应该在控制台打印出SQL查询语句
    spring.jpa.show-sql=true
    
⑦ 项目测试 ( 实际开发中的"问题突显",用 “缓存技术” 能解决这个问题 )
  • 启动项目项目启动成功后,在浏览器上访问 http://localhost:8080/get/1 查询 id为1用户评论信息,此时 假设在浏览器一直刷新访问这个 网址,就会有以下 这种情况:

    浏览器每刷新访问一次服务器端控制台就会输出一条sql语句同时会再执行一次sql操作页面显示还是那一条数据 。( 存在一个实际开发中的问题,这个问题会消耗数据库的性能消耗服务器资源,同时数据库性能下降也会影响 用户的体验 ,这是一个要被解决的问题 )


    在这里插入图片描述

    在这里插入图片描述

  • 之所以出现上面两图情况,这是因为 没有Spring Boot项目开启缓存管理。在没有缓存管理情况下,虽然数据表中数据没有发生变化,但是 每执行 一次查询操作本质执行同样的SQL 语句),都会访问一次数据库并执行一次SQL 语句
    随着时间的积累,系统的用户不断增加数据规模越来越大数据库的操作会直接影响用户的使用体验,此时使用缓存往往是解决这一问题非常好的一种手段

1.2 Spring Boot “默认缓存体验”

  • 前面搭建Web应用 基础上,开启Spring Boot默认支持缓存体验Spring Boot缓存使用效果
(1) 使用 “@EnableCaching” 注解 开始 “缓存管理”
  • 使用 @EnableCaching注解开启 基于 注解缓存支持 ,在项目启动类加入该注解即可 :

    @SpringBootApplication
    @EnableCaching //开启SpringBoot基于注解的"缓存管理"支持
    public class Chapter14Application {
    
        public static void main(String[] args) {
            SpringApplication.run(Chapter14Application.class, args);
        }
    }
    
(2) 使用 “@Cacheable( )” 注解对 “数据操作方法” 进行 “缓存管理” ( 将该注解放在service类的“操作方法”上,让其对"查询结果" 进行 “缓存” )
  • 使用 “@Cacheable ( )注解对 “数据操作方法” 进行 “缓存管理”。 将 @Cacheable( )注解标注Service类查询方法 上,对查询结果进行缓存 :


    在这里插入图片描述


CommentService.java :

package com.myh.chapter_14.service;

import com.myh.chapter_14.Repository.CommentRepository;
import com.myh.chapter_14.domain.Comment;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.cache.annotation.Cacheable;
import org.springframework.stereotype.Controller;
import org.springframework.stereotype.Service;

import java.util.Optional;

/**
 * 使用 @Cacheable()注解开启“缓存管理” : 将查询到的结果存储到“缓存空间”中,下次访问该方法时,不会再去查数据库,而是从“缓存空间”中拿数据
 */
@Service //将该类加入到ioc容器中
public class CommentService { //业务操作类 : 关于操作Comment类相关的业务的CommentService业务类 ( 该类执行业务操作代码 )

    @Autowired
    private CommentRepository commentRepository;

/**
     *  调用CrudRepository接口中的findById()方法来操作数据库
     */
    /*
       ①@Cacheable()注解的作用 :
       添加该注解后,Spring在执行该方法之前会检查缓存中是否有“该查询”的结果,如果有则从其中取出结果,如果没有才去数据库查询
       ②@Cacheable("comment") 中的comment为该"缓存空间"的名称,用区分不同的缓存

       key = "comment_id" 设置该缓存数据的key(“缓存数据”的唯一标识),key属性的值默认为: 方法中参数的值
       key : 表示"缓存数据"的唯一标识
     */
    //
    @Cacheable(cacheNames = "comment",key = "comment_id") //开启“缓存管理”,缓存空间名称为: comment
public Comment findById(int comment_id) { //comment_id的值默认为 该“缓存”的唯一标识( "缓存数据"对应的“key”)
        //调用CrudRepository接口中的findById()方法来操作数据库,有一个返回值类型为Optional<T>类型的对象
        Optional<Comment> optional = commentRepository.findById(comment_id);
        if (optional.isPresent()) { //用于检查Optional对象"是否包含"一个"值"
            return  optional.get(); //返回这个Optional对象包含的"值"
        }
        return null;
    }

}

上述代码中,在CommentService类中的 findByld( int comment_id )方法上添加了 @Cacheable( )注解,该注解作用查询结果 : Comment 存放
Spring Boot默认缓存 中 名称为 : comment名称空间 中。

对应缓存唯一标识 ( 即缓存数据对应的主键 key ) 默认方法参数comment_id的值

(3) Spring Boot默认缓存测试
  • 启动项目,通过浏览器继续访问 “http:/localhost:8080/get/1” 查询id1用户评论信息。此时不论浏览器刷新多少次访问同一个用户评论信息页面的查询结果会显示同一条数据,但**重点**是 后端不用进行多次数据库操作了,此时后端只进行了一次数据库操作


    在这里插入图片描述


    在这里插入图片描述

二、Spring Boot “缓存注解” 介绍 :

  • 上面的内容我们通过使用 @EnableCaching@Cacheable 注解实现了 Spring Boot默认基于注解缓存管理,除此之外,还有更多的 缓存注解以及注解属性可以配置优化缓存管理

2.1 @EnableCaching 注解 ( 用在 “主程序启动类” 上,用于开启SpringBoot的 “缓存管理支持”)

  • @EnableCaching 注解 是由 Spring框架提供的,Spring Boot框架该注解进行了 继承,该注解需要 配置在类上(在Spring Boot中 ,通常 配置项目启动类上),用于 开启基于注解缓存支持

  • @EnableCaching 注解 代码例子如

    Chapter14Application.java ( 主程序启动类 ) :

    @SpringBootApplication
    @EnableCaching //开启SpringBoot的"缓存管理支持"
    public class Chapter14Application {
    
        public static void main(String[] args) {
            SpringApplication.run(Chapter14Application.class, args);
        }
    }
    

2.2 @Cacheable( ) 注解 ( 作用于"类" 或 “方法”,通常用在 “数据查询” 功能的 “方法” 上 , 将 “查询结果” 存进 “缓存空间” 中 )

  • @Cacheable 注解也是由 Spring 框架提供的,可以作用于类或方法通常用数据查询查法 上),用于 方法的查询结果 进行缓存存储 ( 将查询结果 存储在 “缓存空间” 中 )。

    @Cacheable 注解注解一般用在 select 功能方法上 ,当调用该方法时,先根据key去“缓存空间”中查询是否有“缓存数据”,如果,自然就用该“缓存数据” , 如果没有,则去数据库查询数据,然后将该数据 存储到 “缓存空间”中,作为“ 缓存数据”)

  • @Cacheable( )注解执行顺序是,先进行"缓存查询" ( 即在缓存空间查询是否有符合要求数据 ) ,如果 为空进行 “方法查询”如果不为空,则直接使用缓存数据 ( 此时不再进行"方法查询" )

  • @Cacheable( )注解多个属性,用于对“缓存存储”进行相关配置具体属性 及 说明 如下表所示

    属性名说明
    value / cacheNames指定缓存空间名称必配属性。这两个属性 二选一使用
    ps :
    如果 @Cacheable( ) 注解 只有 valuecacheNames( )一个属性时,则该属性的属性名可省略
    key指定 缓存数据key ( 该 “ 缓存”的 唯一标识),默认使用 方法参数值,可以使用SpEL表达式
    keyGenerator指定 缓存数据key生成器与key属性二选一使用
    cacheManager指定 缓存管理器
    cacheResolver指定 缓存解析器,与 cacheManager 属性二选一 使用。
    condition指定符合某条件“进行” 数据缓存
    unless指定符合某条件下“不进行” 数据缓存
    sync指定 是否使用 “异步缓存”默认 false
  • @Cacheable( ) 注解 代码例子如 :

    CommentService.java :

     /**
    
       * 使用 @Cacheable()注解开启“缓存管理” : 将查询到的结果存储到“缓存空间”中,下次访问该方法时,不会再去查数据库,而是从“缓存空间”中拿数据
         */
    @Service //将该类加入到ioc容器中
    public class CommentService { //业务操作类 : 关于操作Comment类相关的业务的CommentService业务类 ( 该类执行业务操作代码 )
    
      @Autowired
      private CommentRepository commentRepository;
    
    /*
     @Cacheable()注解的作用 :
     添加该注解后,Spring在执行该方法之前会检查缓存中是否有“该查询”的结果,如果有则从其中取出结果,如果没有才去数据库查询
    
     @Cacheable("comment") 中的comment为该"缓存空间"的名称,用区分不同的缓存
    */
      @Cacheable(cacheNames = "comment") //开启“缓存管理”,该"缓存空间"名称为: comment
      /*
       如果没有显式地指定该"缓存空间"的key,那么该方法的"参数的值"则为该"key的值",
       所以此处有 comment_id这个参数的值默认为 该“缓存”的唯一标识( "缓存数据"对应的“key”) ,value则是该"缓存值"
       */
      public Comment findById(int comment_id) {
          //调用CrudRepository接口中的findById()方法来操作数据库,有一个返回值类型为Optional<T>类型的对象
          Optional<Comment> optional = commentRepository.findById(comment_id);
          if (optional.isPresent()) { //用于检查Optional对象"是否包含"一个"值"
              return  optional.get(); //返回这个Optional对象包含的"值"
          }
          return null;
       }
     }
    

value / cacheNames 属性 ( 指定存储"缓存数据"的 “空间的名称” )

  • value属性cacheNames 属性 作用相同,用于 指定缓存名称空间 ( 指定“缓存空间”的 名称 ),可以同时指定多个名称空间(例如 @Cacheable ( cacheNames = {“comment1”,“comment2”} ) )。

  • 如果 @Cacheable( ) 注解 只有 valuecacheNames( )一个属性时,则该属性的属性名可省略,例如 @Cacheable(“comment”)指定了缓存的名称空间为 comment

  • @Cacheable( ) 注解的 value / cacheNames 属性代码例子 如 :

    CommentService.java

     /**
    
       * 使用 @Cacheable()注解开启“缓存管理” : 将查询到的结果存储到“缓存空间”中,下次访问该方法时,不会再去查数据库,而是从“缓存空间”中拿数据
         */
    @Service //将该类加入到ioc容器中
    public class CommentService {  
    
      @Autowired
      private CommentRepository commentRepository;
    
      /*
         @Cacheable()注解的作用 :
         添加该注解后,Spring在执行该方法之前会检查缓存中是否有“该查询”的结果,如果有则从其中取出结果,如果没有才去数据库查询
    
         @Cacheable("comment") 中的comment为该"缓存空间"的名称,用区分不同的缓存
       */
      @Cacheable(cacheNames = "comment") //开启“缓存管理”,该"缓存空间"名称为: comment
      public Comment findById(int comment_id) {
          //调用CrudRepository接口中的findById()方法来操作数据库,有一个返回值类型为Optional<T>类型的对象
          Optional<Comment> optional = commentRepository.findById(comment_id);
          if (optional.isPresent()) { //用于检查Optional对象"是否包含"一个"值"
              return  optional.get(); //返回这个Optional对象包含的"值"
          }
          return null;
      }
          //该方法的"查询结果"会被缓存在 comment 和 commentCache 这两个"缓存空间"中
      @Cacheable(cacheNames = {"comment", "commentCache"})
      public Comment findById2(int comment_id) {
          Optional<Comment> optional = commentRepository.findById(comment_id);
          if (optional.isPresent()) {
              return  optional.get();
          }
          return null;
      }
         }
    

key 属性 ( 指定“缓存数据” 对应的 “key / 唯一标识” , 通过这个key就能找到该“缓存数据” ,key属于 和 keyGenerator属性 “二选一” 使用 )

  • key属性作用 是指定缓存数据 对应的 唯一标识 默认使用注解标记方法 中的 “参数” ( 默认使用方法中的参数作为key ), 也可以使用 SpEL表达式

  • 缓存数据本质Map类型数据key 用于 指定唯一标识value用于指定 缓存数据
    如果缓存数据时,没有指定key属性Spring boot默认提供的
    配置类 SimpleKeyGenerator会通过 generateKey( Object…params)方法参数生成key值默认情况下,如果 generateKey( )方法有一个参数参数值就是key属性的值

    如果generateKey( )方法没有参数,那么key属性是一个空参SimpleKey[]对象,如果有多个参数,那么 key属性是一个带参SimpleKey[params1],[param2,…]]对象

  • 除了 使用默认key属性值外,还可以手动指定key属性值,或者是使用Spring框架提供的 SpEL表达式。关于**缓存中支持
    SpEL表达式说明
    下表所示** :

    名称位置描述
    methodNameroot对象当前被调用方法名#root.methodName
    methodroot对象当前被调用方法#root.method.name
    targetroot对象当前被调用的 目标对象实例#root.target
    targetClassroot对象当前被调用的目标对象#root.targetClass
    argsroot 对象当前 被调用的方法参数列表#root.args[0]
    cachesroot对象当前 被调用的方法缓存列表#root.caches[0].name
    Argument执行上下文当前 被调用方法参数,可以用#参数名或者#a0、#p0的形式表示
    ( 0代表参数索引,从0开始 )
    #comment_id、#a0、#p0
    result执行上下文当前方法执行后返回结果#result
  • @Cacheable( ) 注解的 key 属性代码例子 如 :

    CommentService.java

      /**
    
       * 使用 @Cacheable()注解开启“缓存管理” : 将查询到的结果存储到“缓存空间”中,下次访问该方法时,不会再去查数据库,而是从“缓存空间”中拿数据
         */
    @Service //将该类加入到ioc容器中
    public class CommentService2 { //业务操作类
    
      @Autowired
      private CommentRepository commentRepository;
    
      /*
     如果没有显式地指定该"缓存空间"的key,那么该方法的"参数的值"则为该key的值,
     所以此处有 comment_id这个参数的值默认为 该“缓存”的唯一标识( "缓存数据"对应的“key”)
    
     显式地指定该"缓存空间"的key, key为方法中参数的"值",此时value就是该"缓存空间"中的"缓存值"
     */
          /**
       *  1.使用"方法参数" 作为 key
       *
       *  #comment_id 是一个 "SpEL表达式" , 配置该key,意味着每次调用该方法时,都会以key="comment_id的值",来在"缓存空间"中查找 "缓存值"
       */
      @Cacheable(cacheNames = "comment",key = "#comment_id") //显式地指定该"缓存空间"的key为comment_id这个参数的"值"
      public Comment findById(int comment_id) {
          //调用CrudRepository接口中的findById()方法来操作数据库,有一个返回值类型为Optional<T>类型的对象
          Optional<Comment> optional = commentRepository.findById(comment_id);
          if (optional.isPresent()) { //用于检查Optional对象"是否包含"一个"值"
              return  optional.get(); //返回这个Optional对象包含的"值"
          }
          return null;
      }
    
      /**
       *  2.使用 多个 "方法参数" 作为 key
       *
       * "#comment_id + '-' +#userName" :
       * 是一个 "SpEL表达式" , 配置该key,意味着每次调用该方法时,都会以key="#comment_id + '-' +#userName"的值,来在"缓存空间"中查找 "缓存值"
       */
      @Cacheable(cacheNames = "comment",key = "#comment_id + '-' +#userName")
      public Comment findById2(int comment_id,String userName) {
          //调用CrudRepository接口中的findById()方法来操作数据库,有一个返回值类型为Optional<T>类型的对象
          Optional<Comment> optional = commentRepository.findById(comment_id);
          if (optional.isPresent()) { //用于检查Optional对象"是否包含"一个"值"
              return  optional.get(); //返回这个Optional对象包含的"值"
          }
          return null;
      }
        
          /**
       *  3.使用 "方法参数" 集合 "对象的值" 作为 key
       *
       * "#comment_id + '-' +#user.id" :
       * 是一个 "SpEL表达式" , 配置该key,意味着每次调用该方法时,都会以key="#comment_id + '-' +#user.id"的值,来在"缓存空间"中查找 "缓存值"
       */
      @Cacheable(cacheNames = "comment",key = "#comment_id + '-' +#user.id")
      public Comment findById3(int comment_id, User user) {
          //调用CrudRepository接口中的findById()方法来操作数据库,有一个返回值类型为Optional<T>类型的对象
          Optional<Comment> optional = commentRepository.findById(comment_id);
          if (optional.isPresent()) { //用于检查Optional对象"是否包含"一个"值"
              return  optional.get(); //返回这个Optional对象包含的"值"
          }
          return null;
      }
     }
    

CommentController.java

  //@ResponseBody注解 : 将方法的返回值转换为"指定类型",存入响应体中,然后响应给“前端”
  @RestController // 该注解等于 @ResponseBody 注解 + @Controller注解
  public class CommentController {

      @Autowired
      private CommentService commentService;

      @GetMapping("/get/{id}") //路径变量,用@PathVariable()注解来接受路径变量
      public Comment findById(@PathVariable("id") int comment_id) {
          Comment comment = commentService.findById(comment_id);
          return comment;
      }


      @GetMapping("/get/{id}/{userName}") 
      public Comment findById2(@PathVariable("id") int comment_id,@PathVariable("userName") String userName) {
          Comment comment = commentService.findById2(comment_id,userName);
          return comment;
      }


      @GetMapping("/get3/{id}") 
      public Comment findById3(@PathVariable("id") int comment_id, User user) {
          Comment comment = commentService.findById3(comment_id,user);
          return comment;
      }

  }

keyGenerator属性 ( keyGenerator属性 : 指定生成keykey值生成器",该属性key属性二选一使用 )

  • keyGenerator属性kev属性 本质作用 相同,都是用于指定缓存数据key,只不过 keyGenerator 属性指定的不是具休的 key值,而是 key值生成器规则其中 指定的生成器 生成 具体的key 。使用时, kevGenerator属性与key属性要 二者选一。关于自定义key值生成器定义,可以参考Sprina Boot默认配置类SimoleKavGenerator的定义方式。

  • @Cacheable( ) 注解keyGenerator属性代码例子如

    CommentService.java :

      /**
    
       * 使用 @Cacheable()注解开启“缓存管理” : 将查询到的结果存储到“缓存空间”中,下次访问该方法时,不会再去查数据库,而是从“缓存空间”中拿数据
         */
    @Service //将该类加入到ioc容器中
    public class CommentService { //业务操作类
    
      @Autowired
      private CommentKeyGenerator commentKeyGenerator;
    
      /**
       *  通过实现KeyGenerator接口,在CommentKeyGenerator中"自定义"生成key的值
       *  ( 现KeyGenerator接口是属于 keyGenerator属性的内容,该属于与 key属性 "二选一" 使用)
       *  (KeyGenerator属性 和 key属性 "二选一" 使用 )
       */
       @Cacheable(cacheNames = "comment",keyGenerator ="commentKeyGenerator" )  //commentKeyGenerator 为 自定义的 "keyGenerator"
       public Comment findById4(int comment_id,String userName) {
          //调用CrudRepository接口中的findById()方法来操作数据库,有一个返回值类型为Optional<T>类型的对象
          Optional<Comment> optional = commentRepository.findById(comment_id);
          if (optional.isPresent()) { //用于检查Optional对象"是否包含"一个"值"
              return  optional.get(); //返回这个Optional对象包含的"值"
          }
          return null;
        }
      }
    

CommentKeyGenerator.java

/**
 * 在该类中配置key的生成规则
 */
@Component //加入到IOC容器中
public class CommentKeyGenerator implements KeyGenerator {
    @Override
    public Object generate(Object target, Method method, Object... params) {
        int commentId = (int) params[0];
        String userName = (String) params[1];
        return commentId + "_" + userName;
    }
}

cacheManager / cacheResolver属性

  • cacheManager ( 缓存管理器 )和 cacheResolver ( 缓存解析器 )属性分别用于指定 缓存管理器缓存解析器,这两个属性也是

    二选一使用,默认情况不需要配置,如果存在多个cacheManager : 缓存管理器(如 RedisEhcache 等 ),可以 使用这两个属性分别指定

condition 属性 ( 指定条件true时,才对查询结果” 进行 缓存 )

  • condition属性用于对数据进行 有条件的选择性存储,只有当 指定条件true时才会对查询结果进行缓存,可以使用 SpEL表达式指定属性值

    例如@Cacheable(cacheNames = “comment ,condition =”#comment_id>10") 表示方法参数comment_id值大于10 才会对 结果数据进行缓存

  • @Cacheable( ) 注解condition 属性代码例子如

    CommentService.java

    /**
     * 使用 @Cacheable()注解开启“缓存管理” : 将查询到的结果存储到“缓存空间”中,下次访问该方法时,不会再去查数据库,而是从“缓存空间”中拿数据
     */
    @Service //将该类加入到ioc容器中
    public class CommentService { //业务操作类
    
        @Autowired
        private CommentRepository commentRepository;
          /**
       *  @Cacheable()注解中的 condition属性 的使用 : 当指定条件为true时,才会对"查询结果"进行"缓存"
       *  condition = "#comment_id > 3" : 表示当comment_id的值大于3时,才会对相应的数据进行“缓存管理”,否则不会对数据进行"缓存管理"
       */
      @Cacheable(cacheNames = "comment",condition = "#comment_id > 3")
      public Comment findById5(int comment_id) {
          //调用CrudRepository接口中的findById()方法来操作数据库,有一个返回值类型为Optional<T>类型的对象
          Optional<Comment> optional = commentRepository.findById(comment_id);
          if (optional.isPresent()) { //用于检查Optional对象"是否包含"一个"值"
              return  optional.get(); //返回这个Optional对象包含的"值"
          }
          return null;
         }
      }
    

    CommentController.java

    //@ResponseBody注解 : 将方法的返回值转换为"指定类型",存入响应体中,然后响应给“前端”
    @RestController // 该注解等于 @ResponseBody 注解 + @Controller注解
    public class CommentController {
    
        @Autowired
        private CommentService commentService;
    
        @GetMapping("/get5/{id}") //路径变量,用@PathVariable()注解来接受路径变量
        public Comment findById35(@PathVariable("id") int comment_id) {
            Comment comment = commentService.findById5(comment_id);
            return comment;
        }
    
    }
    

unless属性 ( 指定条件false时,才对查询结果” 进行 缓存 )

  • unless属性作用condition属性相反,当指定的条件true 时,方法的返回值不会被缓存 ( 即当 指定条件true时,此时 不会进行“缓存指定条件false,才进行数据的缓存 )。unless属性可以使用 SpEL表达式指定

    例如@Cacheable(cacheNames = "comment ,unless= “#result==nul”) 表示只有查询结果不为空才会对结果数据进行缓存存储

  • @Cacheable( ) 注解unless属性代码例子如

    CommentService.java

    /**
       * 使用 @Cacheable()注解开启“缓存管理” : 将查询到的结果存储到“缓存空间”中,下次访问该方法时,不会再去查数据库,而是从“缓存空间”中拿数据
         */
    @Service //将该类加入到ioc容器中
    public class CommentService2 { //业务操作类
    
      @Autowired
      private CommentRepository commentRepository;
    
      /**
       *  @Cacheable()注解中的 unless 的使用 : 当指定条件为false时,才会对"查询结果"进行"缓存"
       *  cunless = "#comment_id == -1") : 表示当comment_id的值不等于 -1时,才会对相应的数据进行“缓存管理”,否则不会对数据进行"缓存管理"
       */
       @Cacheable(cacheNames = "comment",unless = "#comment_id == -1") //unless : 指定条件为 false才对“查询结果”进行缓存
       public Comment findById6(int comment_id) {
          System.out.println("get6");
          //调用CrudRepository接口中的findById()方法来操作数据库,有一个返回值类型为Optional<T>类型的对象
          Optional<Comment> optional = commentRepository.findById(comment_id);
          if (optional.isPresent()) { //用于检查Optional对象"是否包含"一个"值"
              return  optional.get(); //返回这个Optional对象包含的"值"
          }
          return null;
       }
     }
    

sync属性

sync属性表示数据缓存过程是否使用异步模式默认值false

2.3 @CachePut 注解 ( 作用于"类" 或 “方法”,通常用在 “数据更新” 功能的 “方法” 上 ,在 “更新数据库” 后 “更新缓存” )

  • @CachePut注解是由Spring框型提供的,可以作用于 方法 ( 通常用在"数据更新" 方法 上),该 注解的作用 是 : "更新"缓存数据
    该注解 一般用在 update功能方法 上 ,含义为 : 更新)

    CachePut 注解执行顺序先进行"方法调用" ,然后 将"方法结果" 更新到缓存 中。

    @CachePut注解提供了多个属性,这些属性@Cacheable注解属性完全相同

  • @CachePut 注解 代码例子如

    CommentController.java

    @RestController
    public class CommentController {
    
        @Autowired
        private CommentService commentService;
    
        /**
         *  根据id查询数据 ( 将在service层中添加"缓存管理" )
         */
        @GetMapping("/findCommentById/{id}")
        public Comment findCommentById(@PathVariable("id") Integer id) { //使用@PathVariable路径变量注解来获取“路径下的变量”
            Comment comment = commentService.findCommentById(id);
            return comment;
        }
    
        /**
         * 更新数据
         * 在service层中将 updateComment()方法的"返回值" 更新到"缓存空间"中, 更新"缓存数据" , 这样下次调用 findCommentById()方法的时候就能
         * 获得最新的"缓存空间"的"缓存数据"
         */
        @GetMapping("/updateComment/{id}/{author}")
        public Comment updateComment(@PathVariable("id") Integer id,@PathVariable("author") String author) {
            Comment comment = commentService.updateComment(id,author);
            return comment;
        }
    }
    

    CommentService.java

    @Service //将该类加入到ioc容器中
    public class CommentService {
    
        @Autowired
        private CommentRepository commentRepository;
          /**
       * 使用@CachePut注解来当update数据库后,用该方法的返回值来更新“缓存空间”中的"缓存数据"
       */
      @Cacheable(cacheNames = "comment",key = "#id",condition = "#id > 0")  //id参数的值大于0,才对数据进行"缓存管理"
      public Comment findCommentById(Integer id) {
          Optional<Comment> optional = commentRepository.findById(id);
          //判断Optional是否为空
          if (optional.isPresent()) { //用于检查Optional对象"是否包含"一个"值"
              return  optional.get(); //返回这个Optional对象包含的"值"
          }
          return null;
      }
    
      /**
       * 使用@CachePut()注解来 更新"缓存数据" , 将 updateComment()方法的"返回值" 更新存储到 "缓存空间"中
       *
       * 因为 @CachePut()注解 和 @Cacheable()注解的 “缓存空间名” 和 "key的值" 相同,所以,此处在update数据后,也会及时将“更新数据”存储进“缓存空间”中
       * 所以此时在调用 findCommentById()方法时,会从"缓存空间"中获得“更新后的缓存数据”
       */
      @CachePut(cacheNames = "comment",key = "#id")
      public Comment updateComment(Integer id, String author) {
          Comment comment = commentRepository.findById(id).get();
          comment.setAuthor(author);
         //更新数据
         comment = commentRepository.save(comment);
          return comment;
       }
     }
    

    CommentRepository.java

    package com.myh.chapter_14.Repository;
    
    import com.myh.chapter_14.domain.Comment;
    import org.springframework.data.jpa.repository.JpaRepository;
    
    //Repository接口中为操作数据库的方法
    /*
      继承了JpaRepository接口,其中有操作数据库的curd方法,也用方法关键字的形式来操作数据库,或者使用@Query注解的方式来操作数据库
     */
    //继承JpaRepository接口,从其中获得操作数据库的方法
    public interface CommentRepository extends JpaRepository<Comment,Integer> { 
    
    
    }
    

2.4 @CacheEvict 注解 ( 作用于"类" 或 “方法”,通常用在 “数据删除” 方法上 ,在”删除数据库中 数据“ 后,进行删除 “缓存数据” )

  • @CacheEvict 注解是由Spring框架提供的,可以作用于 方法(通常用在 “数据删除” 方法上),该 注解的作用"删除" 缓存数据

    @CacheEvict 注解默认执行顺序是 : 先进行方法调用然后清除缓存。( 默认先执行方法进行数据库删除然后再清除 “缓存”)

    @CacheEvict注解提供了多个属性,这些属性@Cacheable注解属性基本相同除此之外@CacheEvic注解 额外提供了两个特殊属性 : allEntriesbeforelnvocation其说明如下 :


    (1) allEntries属性

    allEntries属性表示是否清除指定缓存空间中的所有缓存数据默认值false ( 即 默认只删除指定key对应缓存数据 )。
    例如 :
    @CacheEvict (cacheNames = “comment” , allEntries = true ) 表示 : 方法执行后删除缓存空间 comment
    所有的数据


    (2 beforelnvocation属性
    beforeInvocation属性表示 是否方法执行之前 进行 缓存清除默认值为 false ( 即 默认执行方法后 进行 缓存清除)。
    例如 : @CacheEvict ( cacheNames = “comment” , beforelnvocation true) 表示方法执行之前 进行 缓存清除

    需要注意的是 :
    如果将
    @CacheEvict 注解beforelnvocation属性
    设置为 true,会 存在一定的弊端

    例如进行数据删除方法中发生了 异常,这会导致实际数据并没有被删除,但是缓存数据被提前清除了。

  • @CacheEvict 注解 代码例子如

    CommentController.java

    @Controller
    public class CommentController {
    
        @Autowired
        private CommentService commentService;
    
        /**
         *  根据id查询数据 ( 将在service层中添加"缓存管理" )
         */
        @GetMapping("/delete/{id}")
        public void delete(@PathVariable("id") Integer id) { //使用@PathVariable路径变量注解来获取“路径下的变量”
          commentService.delete(id);
        }
    }
    

    CommentService.java :

    @Service //将该类加入到ioc容器中
    public class CommentService {
    
        @Autowired
        private CommentRepository commentRepository;
    
        /**
         *  根据id来删除数据
         */
        @CacheEvict(cacheNames = "comment",key = "#id") //删除数据库的数据后,也删除“缓存空间”中的对应的“缓存数据”
        public void delete(Integer id) {
            System.out.println("数据库的数据删除成功,后清除缓存数据");
        }
    }
    

    CommentRepository.java

    public interface CommentRepository extends JpaRepository<Comment,Integer> {
    
    }
    

2.5 @Caching注解 ( 作用于"类" 或 “方法” )

  • 如果 处理复杂规则数据缓存 可以使用 @Caching 注解注解 作用于 ****或者 方法

  • @Caching注解包含 cacheableputevict 三个属性,这三个属性 作用等同于 @Cacheable注解@CachePut注解@CacheEvict注解实例代码如下 :

    /**
    * 使用@Caching注解
    */
    @Caching(
        cacheable = {@Cacheable(cacheNames = "comment", key = "#id")},
        put = {@CachePut(cacheNames = "comment", key = "#result.author")}
    )
    public Comment findCommentWithAuthor(Long id) {
        // ... 方法实现,查找评论和作者
    }
    

2.6 @CacheConfig注解 ( 作用于"类" , 为该类下 的 “缓存注解” 设置 “公共属性” )

  • @CacheConfig注解作用于,主要用于 统筹管理类中所有使用@CacheableCacheEvict@CacheEvict 注解标注方法中的公共属性,这些公共属性包括 : cacheNameskeyGeneratorcacheManagercacheResolver示例代码如下 :

    CommentService.java

      @Service //将该类加入到ioc容器中
      /**
    
       * 统筹该类下的所有使用@Cacheable()注解、@CachePut()注解、@CacheEvict()注解 , 比如此处统筹其下的方法的“缓存空间”都统一设置为"comment"
         */
    @CacheConfig(cacheNames = "comment")
    public class CommentService {  
    
      @Autowired
      private CommentRepository commentRepository;
      @Cacheable(cacheNames = "comment",key = "comment_id") 
      public Comment findById(int comment_id) { 
          Optional<Comment> optional = commentRepository.findById(comment_id);
          if (optional.isPresent()) { 
              return  optional.get();
          }
          return null;
      }
    }
    

上述代码中,CommentService 类上标注了 @CacheConfig注解,同时使用 cacheNames属性缓存空间统一设置为 comment,这样在 该类中所有方法 上使用 缓存注解时 可以省略相应cacheNames属性


需要说明的是 :如果在类上使用了 @CacheConfig 注解定义了某个属性(例如 cacheNames同时又在该类方法中使用缓存注解定义相同的属性,那么该属性值会使用“就近原则”,方法上注解中的属性值为准

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/563073.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Redis入门到通关之数据结构解析-QuickList

文章目录 ☃️前提概要☃️ 配置项相关☃️简要源码☃️总结 Redis中的 QuickList 是一种特殊的数据结构&#xff0c;用于存储列表类型的数据。它的设计目的是在内存中高效地存储和操作大量的列表元素&#xff0c;尤其是当列表长度很大时。 QuickList的内部结构是一个由多个节…

政安晨:【Keras机器学习示例演绎】(八)—— 利用 PointNet 进行点云分割

目录 简介 导入 下载数据集 加载数据集 构建数据集 预处理 创建 TensorFlow 数据集 PointNet 模型 排列不变性 变换不变性 点之间的相互作用 实例化模型 训练 直观了解培训情况 推论 最后说明 政安晨的个人主页&#xff1a;政安晨 欢迎 &#x1f44d;点赞✍评论…

【PCL】教程 implicit_shape_model.cpp 3D点云数据的对象识别 利用隐式形状模型进行训练和识别...

ism_test_cat.pcd 参数&#xff1a;ism_train_cat.pcd 0 ism_train_horse.pcd 1 ism_train_lioness.pcd 2 ism_train_michael.pcd 3 ism_train_wolf.pcd 4 ism_test_cat.pcd 0 这里红点表示对应感兴趣类别的对象预测中心 ./ism_t…

字节FE:JavaScript学习路线图

JavaScript简介 JavaScript是一种高级的、解释执行的编程语言。它是互联网的三大核心技术之一&#xff0c;与HTML和CSS一同工作&#xff0c;用于创建交互式的网页。JavaScript被所有现代网页浏览器支持而不需要任何插件。它可以增强用户界面和网页的交互性&#xff0c;可以进行…

【讲解下Spring Boot单元测试】

&#x1f308;个人主页: 程序员不想敲代码啊 &#x1f3c6;CSDN优质创作者&#xff0c;CSDN实力新星&#xff0c;CSDN博客专家 &#x1f44d;点赞⭐评论⭐收藏 &#x1f91d;希望本文对您有所裨益&#xff0c;如有不足之处&#xff0c;欢迎在评论区提出指正&#xff0c;让我们共…

FineBi中创建自定义的图表

FineBi中增加自己的自定义图表组件,比如: 的相关笔记: 1 获取有哪些BI自定义图表组件:http://localhost:8080/webroot/decision/v5/plugin/custom/component/list?_=1713667435473[{"name": "图表DEMO_EK","chartType": "amap_demo&q…

GO环境及入门案例

文章目录 简介一、win GO开发环境安装二、Linux go运行环境二、GO代码入门2.1 导包案例2.2 赋值2.3 变量、函数2.4 三方库使用 简介 go不是面向对象语言&#xff0c; 其指针、结构体等比较像C&#xff0c;知名的go 开源项目有docker k8s prometheus node-exporter等 一、win …

如何在3dMax中快速打包mzp 文件?

如何在3dMax中创建mzp 文件&#xff1f; 我喜欢将我的Maxscript脚本发布为mzp文件。这是一个为3dMax构建的自解压zip文件。在mzp文件中&#xff0c;您可以捆绑Maxscript脚本文件、图片、预设或其他文件&#xff0c;并链接安装时执行的特殊操作。 在3dMax中使用大型脚本时&…

耐高温300度锅炉轴承,江苏鲁岳轴承制造的行业标杆

自润滑轴承-产品类型-耐高温轴承-不锈钢轴承-江苏鲁岳轴承制造有限公司。锅炉轴承&#xff0c;耐高温至200度-800度。 江苏鲁岳轴承制造有限公司&#xff0c;一家专注于锅炉轴承和耐高温轴承的研发与生产的企业&#xff0c;致力于为客户提供高质量、高性能的轴承解决方案。其中…

LeetCode题练习与总结:矩阵置零--73

一、题目描述 给定一个 m x n 的矩阵&#xff0c;如果一个元素为 0 &#xff0c;则将其所在行和列的所有元素都设为 0 。请使用 原地 算法。 示例 1&#xff1a; 输入&#xff1a;matrix [[1,1,1],[1,0,1],[1,1,1]] 输出&#xff1a;[[1,0,1],[0,0,0],[1,0,1]]示例 2&#xf…

Linux-内存文件

1. 基础IO操作 1.1 c语言的IO接口 fopen&#xff1a;打开一个文件&#xff0c;按照指定方式 参数&#xff1a;filename 文件名&#xff0c;也可以是路径&#xff0c;mode&#xff1a;打开方式 返回打开的文件指针 fread&#xff1a;从指定流中读数据 参数&#xff1a;从FIL…

Selenium web自动化测试环境搭建

Selenium web自动化环境搭建主要要经历以下几个步骤&#xff1a; 1、安装python 在python官网&#xff1a;Welcome to Python.org&#xff0c;根据各自对应平台如&#xff1a;windows&#xff0c;下载相应的python版本。 ​ 下载成功后&#xff0c;点击安装包&#xff0c;一直…

解释一下“暂存区”的概念,在Git中它扮演什么角色?

文章目录 暂存区在Git中的概念与作用什么是暂存区&#xff08;Staging Area&#xff09;暂存区的位置和结构 暂存区在Git工作流程中的角色1. 分离工作区与版本库的交互示例代码与操作步骤示例1&#xff1a;将工作区的修改添加至暂存区 2. 控制提交内容的粒度示例2&#xff1a;分…

【Linux】虚拟机与Xshell及VS Code的连接

一、基础环境 虚拟机&#xff1a;VMware Workstation Pro 虚拟机镜像&#xff1a;ubuntu-18.04.5-desktop-amd64.iso 其他&#xff1a;Xshell 6、Xftp 6、Visual Studio Code 上述软件的安装操作不再赘述&#xff0c;CSDN上有大量的优秀博文&#xff0c;可参考&#xff1a;详细…

安装和部署maven

准备工作 maven下载地址&#xff1a;https://maven.apache.org/download.cgi 使用wget将maven包下载到linux环境上&#xff0c;/toos/ 目录下&#xff08;也可用迅雷&#xff09; wget https://dlcdn.apache.org/maven/maven-3/3.9.6/binaries/apache-maven-3.9.6-bin.tar.g…

小游戏贪吃蛇的实现之C语言版

找往期文章包括但不限于本期文章中不懂的知识点&#xff1a; 个人主页&#xff1a;我要学编程(ಥ_ಥ)-CSDN博客 所属专栏&#xff1a;C语言 目录 游戏前期准备&#xff1a; 设置控制台相关的信息 GetStdHandle GetConsoleCursorInfo SetConsoleCursorInfo SetConsoleCu…

VSCode插件开发学习

一、环境准备 0、参考文档&#xff1a;VS Code插件创作中文开发文档 1、大于18版本的nodejs 2、安装Yeoman和VS Code Extension Generator&#xff1a; npm install -g yo generator-code 3、生成脚手架 yo code 选择内容&#xff1a; ? What type of extension do yo…

GPT-3.5 Turbo 的 temperature 设置为 0 就是贪婪解码?

&#x1f349; CSDN 叶庭云&#xff1a;https://yetingyun.blog.csdn.net/ 将 GPT-3.5 Turbo 的 temperature 设置为 0 通常意味着采用贪婪解码&#xff08;greedy decoding&#xff09;策略。在贪婪解码中&#xff0c;模型在每一步生成文本时选择概率最高的词元&#xff0c;从…

微调Llama3实践并基于Llama3构建心理咨询EmoLLM

Llama3 Xtuner微调Llama3 EmoLLM 心理咨询师

LabVIEW多设备控制与数据采集系统

LabVIEW多设备控制与数据采集系统 随着科技的进步&#xff0c;自动化测试与控制系统在工业、科研等领域的应用越来越广泛。开发了一种基于LabVIEW平台开发的多设备控制与数据采集系统&#xff0c;旨在解决多设备手动设置复杂、多路数据显示不直观、数据存储不便等问题。通过RS…