Spring | Spring5学习笔记#yyds干货盘点#

风流不在谈锋胜,袖手无言味最长。这篇文章主要讲述Spring | Spring5学习笔记#yyds干货盘点#相关的知识,希望能为你提供帮助。
Spring5学习笔记 0 总结:

  1. Spring框架概述
    (1)轻量级开源javaEE框架,为了解决企业复杂性,两个核心组成:IOC和AOP
    (2)Spring5.2.6版本
  2. IOC容器
    (1)IOC底层原理(工厂、反射等)
    (2)IOC接口(BeanFactory)
    (3)IOC操作Bean管理(基于xml)
    (4)IOC操作Bean管理(基于注解)
  3. Aop
    (1)AOP底层原理:动态代理,有接口(JDK动态代理),没有接口(CGLIB动态代理)
    (2)术语:切入点、增强(通知)、切面
    (3)基于AspectJ实现AOP操作
  4. JdbcTemplate
    (1)使用JdbcTemplate实现数据库curd操作
    (2)使用JdbcTemplate实现数据库批量操作
  5. 事务管理
    (1)事务概念
    (2)重要概念(传播行为和隔离级别)
    (3)基于注解实现声明式事务管理
    (4)完全注解方式实现声明式事务管理
1 Spring5框架概述
  1. Spring是轻量级的开源的JavaEE框架
  2. Spring可以解决企业应用开发的复杂性
  3. Spring有两个核心部分:IOC和Aop
    (1)IOC:控制反转,把创建对象过程交给Spring进行管理
    (2)Aop:面向切面,不修改源代码进行功能增强
  4. Spring特点
    (1)方便解耦,简化开发
    (2)Aop编程支持
    (3)方便程序测试
    (4)方便和其他框架进行整合
    (5)方便进行事务操作
    (6)降低API开发难度
2 IOC(概念和原理)
  1. 什么是IOC
    (1)控制反转,把对象创建和对象之间的调用过程,交给Spring进行管理
    (2)使用IOC目的:为了耦合度降低
    (3)做入门案例就是IOC实现
  2. IOC底层原理
    (1)xml解析、工厂模式、反射
2.1 IOC(BeanFactory接口)
  1. IOC思想基于IOC容器完成,IOC容器底层就是对象工厂
  2. Spring提供IOC容器实现两种方式:(两个接口)
    (1)BeanFactory:IOC容器基本实现,是Spring内部的使用接口,不提供开发人员进行使用
    (2)ApplicationContext:BeanFactory接口的子接口,提供更多更强大的功能,一般由开发人员进行使用
    @Test public void testBean3() ApplicationContext context = new ClassPathXmlApplicationContext("bean4.xml"); //ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("bean4.xml"); Orders orders = context.getBean("orders", Orders.class);

  3. ApplicationContext接口有实现类
    (1)FileSystemXmlApplicationContext
    (2)ClassPathXmlApplicationContext
2.2 IOC操作Bean管理(概念)
  1. 什么是Bean管理
  2. Bean管理操作有两种方式
?(1)基于xml配置文件方式实现
?(2)基于注解方式实现
2.3 IOC操作Bean管理(基于xml方式)
  1. 基于xml方式创建对象
    < !--1 配置User对象创建--> < bean id="user" class="com.atguigu.spring5.User"> < /bean>

    (1)在spring配置文件中,使用bean标签,标签里面添加对应属性,就可以实现对象创建
    (2)在bean标签有很多属性,常用的属性
    > **`id`**属性:唯一标识

    (3)创建对象时候,默认也是执行 < u> 无参数构造< /u> 方法完成对象创建
  2. 基于xml方式注入属性
    (1)DI:依赖注入,就是注入属性
    (2)DI是IOC的一种具体实现,表示依赖注入,或者说注入属性,但需要在创建对象的基础之上完成
  3. 第一种注入方式:使用set方法进行注入
    (1)创建类,定义属性和对应的set方法
    /** * 演示使用set方法进行注入属性 */ public class Book //创建属性 private String bname; private String bauthor; //创建属性对应的set方法 public void setBname(String bname) this.bname = bname; public void setBauthor(String bauthor) this.bauthor = bauthor;

    (2)在spring配置文件配置对象创建,配置属性注入,使用property完成属性注入
    • name:类里面属性名称
    • value:向属性注入的值
    < !--2 set方法注入属性--> < bean id="book" class="com.atguigu.spring5.Book"> < property name="bname" value="https://www.songbingjia.com/android/易筋经"> < /property> < property name="bauthor" value="https://www.songbingjia.com/android/达摩老祖"> < /property> < /bean>

  4. 第二种注入方式:使用有参数构造进行注入
    (1)创建类,定义属性,创建属性对应 < u> 有参数构造< /u> 方法
    /** * 使用有参数构造注入 */ public class Orders //属性 private String oname; private String address; //有参数构造 public Orders(String oname,String address) this.oname = oname; this.address = address;

    (2)在spring配置文件中进行配置
    < !--3 有参数构造注入属性--> < bean id="orders" class="com.atguigu.spring5.Orders"> < constructor-arg name="oname" value="https://www.songbingjia.com/android/电脑"> < /constructor-arg> < constructor-arg name="address" value="https://www.songbingjia.com/android/China"> < /constructor-arg> < /bean>

  5. p名称空间注入(了解)
    (1)第一步添加p名称空间在配置文件中
    < beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:p="http://www.springframework.org/schema/p" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd">

    (2)第二步进行属性注入,在bean标签里面进行操作
    • p:bname
    • p:bauthor
    < !--2 set方法注入属性--> < bean id="book" class="com.atguigu.spring5.Book" p:bname="九阳神功" p:bauthor="无名氏"> < /bean>

2.4 IOC操作Bean管理(xml注入其他类型属性)
  1. 字面量
    (1)null值
    < !--null值--> < property name="address"> < null/> < /property>

    (2)属性值包含特殊符号
    • 把< > 进行转义 & lt; & gt;
    • 把带特殊符号内容写到CDATA
    < property name="address"> < value> < ![CDATA[< < 南京> > ]]> < /value> < /property>

  2. 注入属性-外部bean
    (1)创建service类与dao类
    (2)在service中调用dao里的方法
    (3)在Spring配置文件中进行配置
    public class UserService //创建UserDao类型属性,生成set方法 private UserDao userDao; public void setUserDao(UserDao userDao) this.userDao = userDao; public void add() System.out.println("service add..............."); userDao.update();

    • name属性:类里面属性名称
    • ref属性:创建userDao对象bean标签id值
    < !--1 service和dao对象创建--> < bean id="userService" class="com.atguigu.spring5.service.UserService"> < !--注入userDao对象--> < property name="userDao" ref="userDaoImpl"> < /property> < /bean> < bean id="userDaoImpl" class="com.atguigu.spring5.dao.UserDaoImpl"> < /bean>

  3. 注入属性-内部bean
    (1)一对多关系:部门和员工
    (2)在实体类之间表示一对多关系,员工表示所属部门,使用对象类型属性进行表示
    //部门类 public class Dept private String dname; public void setDname(String dname) this.dname = dname;

    //员工类 public class Emp private String ename; private String gender; //员工属于某一个部门,使用对象形式表示 private Dept dept; public void setDept(Dept dept) this.dept = dept; public void setEname(String ename) this.ename = ename; public void setGender(String gender) this.gender = gender;

    (3)在spring配置文件中进行配置
    < !--内部bean--> < bean id="emp" class="com.atguigu.spring5.bean.Emp"> < !--设置两个普通属性--> < property name="ename" value="https://www.songbingjia.com/android/lucy"> < /property> < property name="gender" value="https://www.songbingjia.com/android/女"> < /property> < !--设置对象类型属性--> < property name="dept"> < bean id="dept" class="com.atguigu.spring5.bean.Dept"> < property name="dname" value="https://www.songbingjia.com/android/安保部"> < /property> < /bean> < /property> < /bean>

  4. 注入属性-级联赋值
    (1)第一种写法
    < !--级联赋值--> < bean id="emp" class="com.atguigu.spring5.bean.Emp"> < !--设置两个普通属性--> < property name="ename" value="https://www.songbingjia.com/android/lucy"> < /property> < property name="gender" value="https://www.songbingjia.com/android/女"> < /property> < !--级联赋值--> < property name="dept" ref="dept"> < /property> < /bean> < bean id="dept" class="com.atguigu.spring5.bean.Dept"> < property name="dname" value="https://www.songbingjia.com/android/财务部"> < /property> < /bean>

    (2)第二种写法
    //员工属于某一个部门,使用对象形式表示 private Dept dept; //生成dept的get方法 public Dept getDept() return dept;

    < !--级联赋值--> < bean id="emp" class="com.atguigu.spring5.bean.Emp"> < !--设置两个普通属性--> < property name="ename" value="https://www.songbingjia.com/android/lucy"> < /property> < property name="gender" value="https://www.songbingjia.com/android/女"> < /property> < !--级联赋值--> < property name="dept" ref="dept"> < /property> < property name="dept.dname" value="https://www.songbingjia.com/android/技术部"> < /property> < /bean> < bean id="dept" class="com.atguigu.spring5.bean.Dept"> < property name="dname" value="https://www.songbingjia.com/android/财务部"> < /property> < /bean>

2.5 IOC操作Bean管理(xml注入集合属性)
  1. 注入数组类型属性
  2. 注入List集合类型属性
  3. 注入Map集合类型属性
    (1)创建类,定义数组、list、map、set类型属性,生成对应set方法
    public class Stu //1 数组类型属性 private String[] courses; //2 list集合类型属性 private List< String> list; //3 map集合类型属性 private Map< String,String> maps; //4 set集合类型属性 private Set< String> sets; public void setSets(Set< String> sets) this.sets = sets; public void setCourses(String[] courses) this.courses = courses; public void setList(List< String> list) this.list = list; public void setMaps(Map< String, String> maps) this.maps = maps;

    (2)在spring配置文件进行配置
    < !--1 集合类型属性注入--> < bean id="stu" class="com.atguigu.spring5.collectiontype.Stu"> < !--数组类型属性注入--> < property name="courses"> < array> < value> java课程< /value> < value> 数据库课程< /value> < /array> < /property> < !--list类型属性注入--> < property name="list"> < list> < value> 张三< /value> < value> 小三< /value> < /list> < /property> < !--map类型属性注入--> < property name="maps"> < map> < entry key="JAVA" value="https://www.songbingjia.com/android/java"> < /entry> < entry key="php" value="https://www.songbingjia.com/android/php"> < /entry> < /map> < /property> < !--set类型属性注入--> < property name="sets"> < set> < value> mysql< /value> < value> Redis< /value> < /set> < /property> < /bean>

  4. 在集合里面设置对象类型值
    < !--注入list集合类型,值是对象--> < property name="courseList"> < list> < ref bean="course1"> < /ref> < ref bean="course2"> < /ref> < /list> < /property> < /bean> < !--创建多个course对象--> < bean id="course1" class="com.atguigu.spring5.collectiontype.Course"> < property name="cname" value="https://www.songbingjia.com/android/Spring5框架"> < /property> < /bean> < bean id="course2" class="com.atguigu.spring5.collectiontype.Course"> < property name="cname" value="https://www.songbingjia.com/android/MyBatis框架"> < /property> < /bean>

  5. 把集合注入部分提取出来
    (1)在spring配置文件中引入名称空间util
    < beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:p="http://www.springframework.org/schema/p" xmlns:util="http://www.springframework.org/schema/util" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/util http://www.springframework.org/schema/util/spring-util.xsd">

    (2)使用util标签完成list集合注入提取
    < !--1 提取list集合类型属性注入--> < util:list id="bookList"> < value> 易筋经< /value> < value> 九阴真经< /value> < value> 九阳神功< /value> < /util:list> < !--2 提取list集合类型属性注入使用--> < bean id="book" class="com.atguigu.spring5.collectiontype.Book" scope="prototype"> < property name="list" ref="bookList"> < /property> < /bean>

2.6 IOC操作Bean管理(FactoryBean)
  1. 普通bean:在配置文件中定义bean类型就是返回类型
  2. 工厂bean:在配置文件定义bean类型可以和返回类型不一样
    (1)第一步创建类,让这个类作为工厂bean,实现接口FactoryBean
    (2)第二步实现接口里面的方法,在实现的方法中定义返回的bean类型
    public class MyBean implements FactoryBean< Course> //定义返回bean @Override public Course getObject() throws Exception Course course = new Course(); course.setCname("abc"); return course; @Override public Class< ?> getObjectType() return null; @Override public boolean isSingleton() return false;

    < bean id="myBean" class="com.at.guigu.spring5.factorybean.MyBean"> < /bean>

2.7 IOC操作Bean管理(bean作用域)
  1. 在Spring里面,设置创建bean实例是单实例还是多实例
  2. 在Spring里面,默认情况下,bean是单实例对象
  3. 如何设置单实例还是多实例
    (1)在spring配置文件bean标签里面有属性(scope)用于设置单实例还是多实例
    (2)scope属性值
    • 第一个值默认值,singleton,表示是单实例对象
    • 第二个值prototype,表示是多实例对象
    < bean id="book" class="com.atguigu.spring5.collectiontype.Book" scope="prototype"> < property name="list" ref="bookList"> < /property> < /bean>

    (3)singletonprototype区别
    • singleton单实例,prototype多实例
    • 设置scope值是singleton时候,加载spring配置文件时候就会创建单实例对象
    • 设置scope值是prototype时候,不是在加载spring配置文件时候创建对象,在调用getBean方法时候创建多实例对象
2.8 IOC操作Bean管理(bean生命周期)
  1. 生命周期
    (1)从对象创建到对象销毁的过程
  2. bean生命周期
    (1)通过构造器创建bean实例(无参数构造)
    (2)为bean的属性设置值和对其他bean引用(调用set方法)
    (3)调用bean的初始化的方法(需要进行配置初始化的方法)
    (4)bean可以使用了(对象获取到了)
    (5)当容器关闭时候,调用bean的销毁的方法(需要进行配置销毁的方法)
  3. 演示bean生命周期
    public class Orders //无参数构造 public Orders() System.out.println("第一步 执行无参数构造创建bean实例"); private String oname; public void setOname(String oname) this.oname = oname; System.out.println("第二步 调用set方法设置属性值"); //创建执行的初始化的方法 public void initMethod() System.out.println("第三步 执行初始化的方法"); //创建执行的销毁的方法 public void destroyMethod() System.out.println("第五步 执行销毁的方法");

    < bean id="orders" class="com.atguigu.spring5.bean.Orders" init-method="initMethod" destroy-method="destroyMethod"> < property name="oname" value="https://www.songbingjia.com/android/手机"> < /property> < /bean>

    @Test public void testBean3() ApplicationContext context = new ClassPathXmlApplicationContext("bean4.xml"); Orders orders = context.getBean("orders", Orders.class); System.out.println("第四步 获取创建bean实例对象"); System.out.println(orders); //手动让bean实例销毁 context.close();

  4. bean的后置处理器,bean生命周期有七步
    (1)通过构造器创建bean实例(无参数构造)
    (2)为bean的属性设置值和对其他bean引用(调用set方法)
    (3)把bean实例传递bean后置处理器的方法postProcessBeforeInitialization
    (4)调用bean的初始化的方法(需要进行配置初始化的方法)
    (5)把bean实例传递bean后置处理器的方法postProcessAfterInitialization
    (6)bean可以使用了(对象获取到了)
    (7)当容器关闭时候,调用bean的销毁的方法(需要进行配置销毁的方法)
  5. 演示添加后置处理器效果
    (1)创建类,实现接口BeanPostProcessor,创建后置处理器
    public class MyBeanPost implements BeanPostProcessor @Override public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException System.out.println("在初始化之前执行的方法"); return bean; @Override public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException System.out.println("在初始化之后执行的方法"); return bean;

    (2)配置后置处理器
    < !--配置后置处理器--> < bean id="myBeanPost" class="com.atguigu.spring5.bean.MyBeanPost"> < /bean>

2.9 IOC操作Bean管理(xml自动装配)
  1. 自动装配:根据指定装配规则(属性名称或者属性类型),Spring自动将匹配的属性值进行注入
  2. 自动装配示例
    (1)bean标签属性autowire,配置自动装配
    (2)autowire属性常用两个值:
    • byName根据属性名称注入 ,注入值beanid值和类属性名称一样
    • byType根据属性类型注入
    < bean id="emp" class="com.atguigu.spring5.autowire.Emp" autowire="byName"> < /bean> < bean id="dept" class="com.atguigu.spring5.autowire.Dept"> < /bean>

    < bean id="emp" class="com.atguigu.spring5.autowire.Emp" autowire="byType"> < /bean> < bean id="dept" class="com.atguigu.spring5.autowire.Dept"> < /bean>

2.10 IOC操作Bean管理(外部属性文件)
  1. 直接配置数据库信息
    (1)配置德鲁伊连接池
    (2)引入德鲁伊连接池依赖jar包druid-1.1.9.jar
    < !--直接配置连接池--> < bean id="dataSource" class="com.alibaba.druid.pool.DruidDataSource"> < property name="driverClassName" value="https://www.songbingjia.com/android/com.mysql.jdbc.Driver"> < /property> < property name="url" value="https://www.songbingjia.com/android/jdbc:mysql://localhost:3306/userDb"> < /property> < property name="username" value="https://www.songbingjia.com/android/root"> < /property> < property name="password" value="https://www.songbingjia.com/android/root"> < /property> < /bean>

  2. 引入外部属性文件配置数据库连接池(建议)
    (1)创建外部属性文件,properties格式文件,写数据库信息
    prop.driverClass=com.mysql.jdbc.Driver prop.url=jdbc:mysql://localhost:3306/userDb prop.userName=root prop.password=root

    (2)把外部properties属性文件引入到spring配置文件中
    • 引入context名称空间
      < ?xml version="1.0" encoding="UTF-8"?> < beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:context="http://www.springframework.org/schema/context" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd">

    • 在spring配置文件使用标签引入外部属性文件
      < !--引入外部属性文件--> < context:property-placeholder location="classpath:jdbc.properties"/> < !--配置连接池--> < bean id="dataSource" class="com.alibaba.druid.pool.DruidDataSource"> < property name="driverClassName" value="https://www.songbingjia.com/android/$prop.driverClass"> < /property> < property name="url" value="https://www.songbingjia.com/android/$prop.url"> < /property> < property name="username" value="https://www.songbingjia.com/android/$prop.userName"> < /property> < property name="password" value="https://www.songbingjia.com/android/$prop.password"> < /property> < /bean>

2.11 IOC操作Bean管理(基于注解方式)
  1. 什么是注解
    (1)注解是代码特殊标记,格式:@注解名称(属性名称=属性值,属性名称=属性值..)
    (2)使用注解,注解作用在类上面,方法上面,属性上面
    (3)使用注解目的:简化xml配置
  2. Spring针对Bean管理中创建对象提供注解
    (1)@Repository:数据库层组件,DAO层;(2)**`@Service`**:业务逻辑组件Service层;
    br/>(2)**`@Service`**:业务逻辑组件Service层;
    @Controller:控制器,Controller层;
    (4)@Component:组件,POJO层。
  3. 基于注解方式实现对象创建
    (1)第一步 引入依赖spring-aop-5.2.6.RELEASE.jar
    (2)第二步 开启组件扫描
    • 如果扫描多个包,则多个包用逗号隔开
    • 扫描包上层目录
    < context:component-scan base-package="com.atguigu"> < /context:component-scan>

    (3)第三步 创建类,在类上添加创建对象注解
    • 在注解里面value属性值可以省略不写
    • 默认值是类名称,首字母小写,即UserService -- userService
    @Component(value = "https://www.songbingjia.com/android/userService")//相当于< bean id="userService" class=".."/> public class UserService public void add() System.out.println("service add.......");

  4. 开启组件扫描细节配置
    (1)设置扫描哪些内容:
    • use-default-filters="false"时,表示不使用默认filter,自己设置filter
    • context:include-filter表示设置扫描哪些内容
      < context:component-scan base-package="com.atguigu" use-default-filters="false"> < context:include-filter type="annotation" expression="org.springframework.stereotype.Controller"/> < /context:component-scan>

    (2)设置不扫描哪些内容:
    • context:exclude-filter表示设置不扫描哪些内容
      < context:component-scan base-package="com.atguigu"> < context:exclude-filter type="annotation" expression="org.springframework.stereotype.Controller"/> < /context:component-scan>

  5. 基于注解方式实现属性注入
    (1)@Autowired:根据属性类型进行自动装配
    • 第一步把servicedao对象创建,在servicedao类添加创建对象注解
    • 第二步在service注入dao对象,在service类添加dao类型属性,在属性上面使用注解
    • 类型相当于类名
      @Service public class UserService //定义dao类型属性 //不需要添加set方法 //添加注入属性注解 @Autowired private UserDao userDao; public void add() System.out.println("service add......."); userDao.add();

    (2)@Qualifier:根据名称进行注入(要和@Autowired一起使用)
    • 名称相当于对象名
      @Autowired//根据类型进行注入 @Qualifier(value = "https://www.songbingjia.com/android/userDaoImpl1") //根据名称进行注入 private UserDao userDao;

    (3)@Resourse:可以根据类型注入,也可以根据名称注入
    • 相当于@Autowired@Qualifier
      //@Resource//根据类型进行注入 @Resource(name = "userDaoImpl1")//根据名称进行注入 private UserDao userDao;

    (4)@Value:注入普通类型属性
    • 普通注入
      @Value(value = "https://www.songbingjia.com/android/abc") private String name;

  6. 完全注解开发(SpringBoot)
    • 创建配置类,替代xml配置文件
    • 开启包扫描@ComponentScan(basePackages = "com.atguigu")
      @Configuration//作为配置类,替代xml配置文件 @ComponentScan(basePackages = "com.atguigu") public class SpringConfig

    • 编写测试类
      @Test public void testService2() //加载配置类 ApplicationContext context = new AnnotationConfigApplicationContext(SpringConfig.class); UserService userService = context.getBean("userService", UserService.class); System.out.println(userService); userService.add();

3 AOP(1)面向切面编程(方面),利用AOP可以对业务逻辑的各个部分进行隔离,从而使得业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高了开发的效率。
(2)通俗描述:不通过修改源代码方式,在主干功能里面添加新功能
3.1 AOP(底层原理)
?(1)第一种有接口情况,使用JDK动态代理:创建接口实现类代理对象,增强类的方法
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-CfG8wIuI-1623595927676)(D:\\桌面文件夹\\校内文件\\大二下\\我的Markdowm笔记\\spring5图片\\AOP创建接口JDK动态代理.png)]
?(2)第二种没有接口情况,使用CGLIB动态代理:创建子类的代理对象,增强类的方法
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-YSWOe9Ij-1623595927679)(D:\\桌面文件夹\\校内文件\\大二下\\我的Markdowm笔记\\spring5图片\\AOP创建子类JDK动态代理.png)]
3.2 AOP(JDK动态代理底层原理)
  1. 使用JDK动态代理,使用Proxy类里面的方法创建代理对象
    (1)调用newProxyInstance方法
    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-e8Itd0IA-1623595927681)(D:\\桌面文件夹\\校内文件\\大二下\\我的Markdowm笔记\\spring5图片\\newProxyInstance.png)]
    方法有三个参数:
    • 第一参数,类加载器;
    • 第二参数,增强方法所在的类,这个类实现的接口,支持多个接口;
    • 第三参数,实现这个接口InvocationHandler,创建代理对象,写增强的部分;
  2. 编写JDK动态代理代码
    (1)创建接口,定义方法
    public interface UserDao public int add(int a, int b); public String update(String id);

    (2)创建接口实现类,实现方法
    public class UserDaoImpl implements UserDao @Override public int add(int a, int b) System.out.println("add方法执行了....."); return a+b; @Override public String update(String id) System.out.println("update方法执行了....."); return id;

    (3)使用Proxy类创建接口代理对象
    public class JDKProxy public static void main(String[] args) //创建接口实现类代理对象 Class[] interfaces = UserDao.class; UserDaoImpl userDao = new UserDaoImpl(); UserDao dao = (UserDao)Proxy.newProxyInstance(JDKProxy.class.getClassLoader(), interfaces, new UserDaoProxy(userDao)); int result = dao.add(1, 2); System.out.println("result:"+result); //创建代理对象代码 class UserDaoProxy implements InvocationHandler //把创建的是谁的代理对象,把谁传递过来 //有参数构造传递 private Object obj; public UserDaoProxy(Object obj) this.obj = obj; //增强的逻辑 @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable //方法之前 System.out.println("方法之前执行...."+method.getName()+" :传递的参数..."+ Arrays.toString(args)); //被增强的方法执行 Object res = method.invoke(obj, args); //方法之后 System.out.println("方法之后执行...."+obj); return res;

3.3 AOP(术语)
  1. 连接点
    • 类里面哪些方法可以被增强,这些方法称为连接点
  2. 切入点
    • 实际被真正增强的方法
  3. 通知(增强)
    • 实际增强的逻辑部分
    • 通知有多种类型:
  4. 切面
    • 是动词
    • 把通知应用到切入点的过程
3.4 AOP操作(准备工作)
  1. Spring框架一般都是基于AspectJ实现AOP操作
    • AspectJ不是Spring组成部分,独立AOP框架,一般把AspectJ和Spirng框架一起使用,进行AOP操作
  2. 基于AspectJ实现AOP操作
    (1)基于xml配置文件实现
    (2)基于注解方式实现(使用)
  3. 在项目工程里面引入AOP相关依赖
    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-Ts8zfehP-1623595927683)(D:\\桌面文件夹\\校内文件\\大二下\\我的Markdowm笔记\\spring5图片\\AOP相关依赖.png)]
  4. 切入点表达式
    (1)切入点表达式作用:知道对哪个类里面的哪个方法进行增强
    (2)语法结构:execution([权限修饰符] [返回类型] [类全路径] [方法名称]([参数列表]) )
    • 举例1:对com.atguigu.dao.BookDao类里面的add进行增强
      • *`execution( com.atguigu.dao.BookDao.add(..))`**
    • 举例2:对com.atguigu.dao.BookDao类里面的所有的方法进行增强
      • execution(* com.atguigu.dao.BookDao.* (..))
    • 举例3:对com.atguigu.dao包里面所有类,类里面所有方法进行增强
      • *`execution( com.atguigu.dao.. (..))`**
3.5 AOP操作(AspectJ配置文件)
  1. 创建两个类,增强类和被增强类,创建方法
    public class Book public void buy() System.out.println("buy.............");

    public class BookProxy public void before() System.out.println("before.........");

  2. 在spring配置文件中创建两个类对象
    < !--创建对象--> < bean id="book" class="com.atguigu.spring5.aopxml.Book"> < /bean> < bean id="bookProxy" class="com.atguigu.spring5.aopxml.BookProxy"> < /bean>

  3. 在spring配置文件中配置切入点
    < !--配置aop增强--> < aop:config> < !--切入点--> < aop:pointcut id="p" expression="execution(* com.atguigu.spring5.aopxml.Book.buy(..))"/> < !--配置切面--> < aop:aspect ref="bookProxy"> < !--增强作用在具体的方法上--> < aop:before method="before" pointcut-ref="p"/> < /aop:aspect> < /aop:config>

3.6 AOP操作(AspectJ注解)
  1. 创建类,并在类里定义方法
    public class User public void add() System.out.println("add.......");

  2. 创建增强类(编写增强逻辑)
    //增强的类 public class UserProxy public void before()//前置通知 System.out.println("before.........");

  3. 进行通知的配置
    (1)在spring配置文件中,开启注解扫描
    < ?xml version="1.0" encoding="UTF-8"?> < beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:context="http://www.springframework.org/schema/context" xmlns:aop="http://www.springframework.org/schema/aop" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop.xsd"> < !-- 开启注解扫描 --> < context:component-scan base-package="com.atguigu.spring5.aopanno"> < /context:component-scan>

    (2)使用注解创建UserUserProxy对象
    @Component public class User

    @Component public class UserProxy

    (3)在增强类上面添加注解@Aspect
    @Component @Aspect //生成代理对象 public class UserProxy

    (4)在spring配置文件中开启生成代理对象
    < !-- 开启Aspect生成代理对象--> < aop:aspectj-autoproxy> < /aop:aspectj-autoproxy>

  4. 配置不同类型的通知
    • 在增强类的里面,在作为通知方法上面添加通知类型注解,使用切入点表达式配置
    • 可以进行相同的切入点抽取,使用@Pointcut注解
      //增强的类 @Component @Aspect//生成代理对象 public class UserProxy //相同切入点抽取 @Pointcut(value = "https://www.songbingjia.com/android/execution(* com.atguigu.spring5.aopanno.User.add(..))") public void pointdemo() //前置通知 //@Before注解表示作为前置通知 @Before(value = "https://www.songbingjia.com/android/pointdemo()") public void before() System.out.println("before........."); //后置通知(返回通知) @AfterReturning(value = "https://www.songbingjia.com/android/execution(* com.atguigu.spring5.aopanno.User.add(..))") public void afterReturning() System.out.println("afterReturning........."); //最终通知 @After(value = "https://www.songbingjia.com/android/execution(* com.atguigu.spring5.aopanno.User.add(..))") public void after() System.out.println("after........."); //异常通知 @AfterThrowing(value = "https://www.songbingjia.com/android/execution(* com.atguigu.spring5.aopanno.User.add(..))") public void afterThrowing() System.out.println("afterThrowing........."); //环绕通知 @Around(value = "https://www.songbingjia.com/android/execution(* com.atguigu.spring5.aopanno.User.add(..))") public void around(ProceedingJoinPoint proceedingJoinPoint) throws Throwable System.out.println("环绕之前........."); //被增强的方法执行 proceedingJoinPoint.proceed(); System.out.println("环绕之后.........");

  5. 有多个增强类同一个方法进行增强,设置增强类优先级
    • 在增强类上面添加注解@Order(数字类型值),数字类型值越小优先级越高
      @Component @Aspect @Order(1) public class PersonProxy

  6. 完全使用注解开发
    • 创建配置类,不需要创建xml配置文件
      @Configuration @ComponentScan(basePackages = "com.atguigu") @EnableAspectJAutoProxy(proxyTargetClass = true) public class ConfigAop

4 Spring操作JdbcTemplate 4.1 JdbcTemplate(概念和准备)
  1. 什么是JdbcTemplate
    (1)Spring框架对JDBC进行封装,使用JdbcTemplate方便实现对数据库操作
  2. 准备工作
    (1)引入相关jar包
    • druid-1.1.9.jar
    • mysql-connector-java-5.1.7-bin.jar
    • spring-jdbc-5.2.6.RELEASE.jar
    • spring-orm-5.2.6.RELEASE.jar
    • spring-tx-5.2.6.RELEASE.jar
    (2)在spring配置文件配置数据库连接池
< !-- 数据库连接池 --> < bean id="dataSource"class="com.alibaba.druid.pool.DruidDataSource"destroy-method="close"> < property name="url" value="https://www.songbingjia.com/android/jdbc:mysql:///user_db" /> < property name="username" value="https://www.songbingjia.com/android/root" /> < property name="password" value="https://www.songbingjia.com/android/root" /> < property name="driverClassName"value="https://www.songbingjia.com/android/com.mysql.jdbc.Driver" /> < /bean>

?(3)配置JdbcTemplate对象,注入DataSource
< !-- JdbcTemplate对象 --> < bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate"> < !--注入dataSource--> < property name="dataSource" ref="dataSource"> < /property> < /bean>

?(4)创建service类,创建dao类,在dao注入jdbcTemplate对象
< !-- 组件扫描 --> < context:component-scan base-package="com.atguigu"> < /context:component-scan>

@Service public class BookService //注入dao @Autowired private BookDao bookDao;

@Repository public class UserDaoImpl implements BookDao //注入JdbcTemplate @Autowired private JdbcTemplate jdbcTemplate;

4.2 JdbcTemplate操作数据库(添加)
  1. 对应数据库创建实体类
  2. 编写service和dao
    (1)在dao进行数据库添加操作
    (2)调用JdbcTemplate对象里面update方法实现添加操作
    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-st95Z5j2-1623595927685)(D:\\桌面文件夹\\校内文件\\大二下\\我的Markdowm笔记\\spring5图片\\JdbcTemplate对象里面update方法.png)]有两个参数
    • 第一个参数:sql语句
    • 第二个参数:可变参数,设置sql语句值
      @Repository public class BookDaoImpl implements BookDao //注入 JdbcTemplate @Autowired private JdbcTemplate jdbcTemplate; //添加的方法 @Override public void add(Book book) //1 创建 sql 语句 String sql = "insert into t_book values(?,?,?)"; //2 调用方法实现 Object[] args = book.getUserId(), book.getUsername(),book.getUstatus(); int update = jdbcTemplate.update(sql,args); System.out.println(update);

4.3 JdbcTemplate操作数据库(修改和删除)
//1、修改 @Override public void updateBook(Book book) String sql = "update t_book set username=?,ustatus=? where user_id=?"; Object[] args = book.getUsername(), book.getUstatus(),book.getUserId(); int update = jdbcTemplate.update(sql, args); System.out.println(update); //2、删除 @Override public void delete(String id) String sql = "delete from t_book where user_id=?"; int update = jdbcTemplate.update(sql, id); System.out.println(update); //使用JdbcTemplate 模板所实现的 “增删改” 都是调用了同一个 “update” 方法

4.4 JdbcTemplate操作数据库(查询返回某个值)
  1. 查询表里面有多少条记录,返回是某个值
  2. 使用JdbcTemplate实现查询返回某个值代码
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-hQtgZOQ8-1623595927686)(D:\\桌面文件夹\\校内文件\\大二下\\我的Markdowm笔记\\spring5图片\\queryForObject.png)]
?有两个参数
  • 第一个参数:sql语句
  • 第二个参数:返回类型Class
//查询表记录数 @Override public int selectCount() String sql = "select count(*) from t_book"; Integer count = jdbcTemplate.queryForObject(sql, Integer.class); return count;

4.5 JdbcTemplate操作数据库(查询返回对象)
  1. 场景:查询图书详情
  2. JdbcTemplate实现查询返回对象
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-JQV8LaMR-1623595927687)(D:\\桌面文件夹\\校内文件\\大二下\\我的Markdowm笔记\\spring5图片\\queryForObject2.png)]
?有三个参数
  • 第一个参数:sql语句
  • 第二个参数:RowMapper是接口,针对返回不同类型数据,使用这个接口里面实现类完成数据封装
  • 第三个参数:sql语句值
//查询返回对象 @Override public Book findBookInfo(String id) String sql = "select * from t_book where user_id=?"; //调用方法 Book book = jdbcTemplate.queryForObject(sql, new BeanPropertyRowMapper< Book> (Book.class), id); return book;

4.6 JdbcTemplate操作数据库(查询返回集合)
  1. 场景:查询图书列表分页…
  2. 调用JdbcTemplate方法实现查询返回集合
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-AB2hS1S7-1623595927688)(D:\\桌面文件夹\\校内文件\\大二下\\我的Markdowm笔记\\spring5图片\\query.png)]
有三个参数
  • 第一个参数:sql语句
  • 第二个参数:RowMapper是接口,针对返回不同类型数据,使用这个接口里面实现类完成数据封装
  • 第三个参数:sql语句值
//所用场景:查询图书列表分页、、 //查询返回集合 @Override public List< Book> findAllBook() String sql = "select * from t_book"; //调用方法 List< Book> bookList = jdbcTemplate.query(sql, new BeanPropertyRowMapper< Book> (Book.class)); return bookList;

4.7 JdbcTemplate操作数据库(批量操作)
  1. 批量操作:操作表里面多条记录
  2. JdbcTemplate实现批量添加操作
    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-7pEz81bi-1623595927689)(D:\\桌面文件夹\\校内文件\\大二下\\我的Markdowm笔记\\spring5图片\\batchUpdate.png)]
    有两个参数
    • 第一个参数:sql语句
    • 第二个参数:List集合,添加多条记录数据
      @Override public void batchAddBook(List< Object[]> batchArgs) String sql = "insert into t_book values(?,?,?)"; int[] ints = jdbcTemplate.batchUpdate(sql, batchArgs); System.out.println(Arrays.toString(ints)); //批量添加测试 List< Object[]> batchArgs = new ArrayList< > (); Object[] o1 = "3","java","a"; Object[] o2 = "4","c++","b"; Object[] o3 = "5","MySQL","c"; batchArgs.add(o1); batchArgs.add(o2); batchArgs.add(o3); //调用批量添加 bookService.batchAdd(batchArgs);

  3. JdbcTemplate实现批量修改操作
    //批量修改(同批量添加一样,调用同一个方法) @Override public void batchUpdateBook(List< Object[]> batchArgs) String sql = "update t_book set username=?,ustatus=? where user_id=?"; int[] ints = jdbcTemplate.batchUpdate(sql, batchArgs); System.out.println(Arrays.toString(ints)); //批量修改测试 List< Object[]> batchArgs = new ArrayList< > (); Object[] o1 = "java","a","3"; Object[] o2 = "c++","b","5"; Object[] o3 = "MySQL","c","6"; batchArgs.add(o1); batchArgs.add(o2); batchArgs.add(o3); //调用批量添加 bookService.batchUpdate(batchArgs);

  4. JdbcTemplate实现批量删除操作
    //批量删除 @Override public void batchDeleteBook(List< Object[]> batchArgs) String sql = "delete from t_book where user_id=?"; int[] ints = jdbcTemplate.batchUpdate(sql, batchArgs); System.out.println(Arrays.toString(ints)); //批量删除测试 List< Object[]> batchArgs = new ArrayList< > (); Object[] o1 = "3"; Object[] o2 = "5"; batchArgs.add(o1); batchArgs.add(o2); batchArgs.add(o3); //调用批量添加 bookService.batchDelete(batchArgs);

5 事务 5.1 事务操作(事务概念)
  1. 什么事务
    (1)事务是数据库操作最基本单元,逻辑上一组操作,要么都成功,如果有一个失败所有操作都失败
    (2)典型场景:银行转账
  • lucy转账100元给mary
  • lucy少100,mary多100
  1. 事务四个特性(ACID)
    (1)原子性
    (2)一致性
    (3)隔离性
    (4)持久性
5.2 事务操作(搭建事务操作环境)
  1. 创建数据库表,添加记录
  2. 创建service,搭建dao,完成对象创建和注入关系
    (1)service注入dao,在dao注入JdbcTemplate,在JdbcTemplate注入DataSource
    @Service public class UserService //注入dao @Autowired private UserDao userDao;

    @Repository public class UserDaoImpl implements UserDao //注入JdbcTemplate @Autowired private JdbcTemplate jdbcTemplate;

  3. 在dao创建两个方法:多钱和少钱的方法,在service创建方法(转账的方法),并模拟异常
    @Repository public class UserDaoImpl implements UserDao @Autowired private JdbcTemplate jdbcTemplate; //lucy转账100给mary //少钱 @Override public void reduceMoney() String sql = "update t_account set money=money-? where username=?"; jdbcTemplate.update(sql,100,"lucy"); //多钱 @Override public void addMoney() String sql = "update t_account set money=money+? where username=?"; jdbcTemplate.update(sql,100,"mary");

    @Service public class UserService //注入dao @Autowired private UserDao userDao; //转账的方法 public void accountMoney() //lucy少100 userDao.reduceMoney(); //模拟异常 int i = 10/0; //mary多100 userDao.addMoney();

5.3 事务操作(Spring事务管理介绍)
  1. 事务添加到JavaEE三层结构里面Service层(业务逻辑层)
  2. 在Spring进行事务管理操作
    (1)有两种方式:编程式事务管理和声明式事务管理(使用)
  3. 声明式事务管理
    (1)基于注解方式(使用)
    (2)基于xml配置文件方式
  4. 在Spring进行声明式事务管理,底层使用AOP原理
  5. Spring事务管理API
    (1)提供一个接口,代表事务管理器,这个接口针对不同的框架提供不同的实现类
    DataSourceTransactionManager(org.springframework.jdbc.datasource)
5.4 事务操作(注解声明式事务管理)
  1. 在spring配置文件配置事务管理器
    < !--创建事务管理器--> < bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager"> < !--注入数据源--> < property name="dataSource" ref="dataSource"> < /property> < /bean>

  2. 在spring配置文件,开启事务注解
    (1)在spring配置文件引入名称空间tx
    < ?xml version="1.0" encoding="UTF-8"?> < beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:context="http://www.springframework.org/schema/context" xmlns:aop="http://www.springframework.org/schema/aop" xmlns:tx="http://www.springframework.org/schema/tx" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop.xsd http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx.xsd">

    (2)开启事务注解
    < !--开启事务注解--> < tx:annotation-driven transaction-manager="transactionManager"> < /tx:annotation-driven>

  3. 在service类上面(或者service类里面方法上面)添加事务注解
    (1)@Transactional,这个注解添加到类上面,也可以添加方法上面
    (2)如果把这个注解添加类上面,这个类里面所有的方法都添加事务
    (3)如果把这个注解添加方法上面,为这个方法添加事务
    @Service @Transactional public classUserService

5.5 事务操作(声明式事务管理参数配置)
  1. 在service类上面添加注解@Transactional,在这个注解里面可以配置事务相关参数
    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-4PY8x1y5-1623595927690)(D:\\桌面文件夹\\校内文件\\大二下\\我的Markdowm笔记\\spring5图片\\Transactional注解.png)]
  2. propagation:事务传播行为
    (1)多事务方法直接进行调用,这个过程中事务是如何进行管理的
    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-lPm6NzFD-1623595927691)(D:\\桌面文件夹\\校内文件\\大二下\\我的Markdowm笔记\\spring5图片\\事务传播行为.png)]
    【Spring | Spring5学习笔记#yyds干货盘点#】(2)Spring定义了7种类传播行为
    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-eEIFuQQQ-1623595927692)(D:\\桌面文件夹\\校内文件\\大二下\\我的Markdowm笔记\\spring5图片\\事务传播行为属性.png)]
    @Service @Transactional(propagation = Propagation.REQUIRED) public class UserService

  3. ioslation:事务隔离级别
    (1)事务有特性成为隔离性,多事务操作之间不会产生影响。不考虑隔离性产生很多问题
    (2)有三个读问题:脏读、不可重复读、虚(幻)读
    (3)脏读:一个未提交事务读取到另一个未提交事务的数据
    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-h8khCrBW-1623595927693)(D:\\桌面文件夹\\校内文件\\大二下\\我的Markdowm笔记\\spring5图片\\脏读.png)]
    (4)不可重复读:一个未提交事务读取到另一提交事务修改数据
    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-0GQKfNc8-1623595927694)(D:\\桌面文件夹\\校内文件\\大二下\\我的Markdowm笔记\\spring5图片\\不可重复读.png)]
    (5)虚读:一个未提交事务读取到另一提交事务添加数据
    (6)解决:通过设置事务隔离级别,解决读问题
    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-KaeuhXnR-1623595927695)(D:\\桌面文件夹\\校内文件\\大二下\\我的Markdowm笔记\\spring5图片\\事务隔离级别解决.png)]
    @Service @Transactional(propagation = Propagation.REQUIRED,isolation = Isolation.REPEATABLE_READ) public class UserService

  4. timeout:超时时间
    (1)事务需要在一定时间内进行提交,如果不提交进行回滚
    (2)默认值是-1,设置时间以秒单位进行计算
  5. readOnly:是否只读
    (1)读:查询操作,写:添加修改删除操作
    (2)readOnly默认值false,表示可以查询,可以添加修改删除操作
    (3)设置readOnly值是true,设置成true之后,只能查询
  6. rollbackFor:回滚
    (1)设置出现哪些异常进行事务回滚
  7. noRollbackFor:不回滚
    (1)设置出现哪些异常不进行事务回滚
5.6 事务操作(XML声明式事务管理)
  • 在spring配置文件中进行配置
    (1)第一步配置事务管理器
    (2)第二步配置通知
    (3)配置切入点与切面
    < !--1 创建事务管理器--> < bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager"> < !--注入数据源--> < property name="dataSource" ref="dataSource"> < /property> < /bean> < !--2 配置通知--> < tx:advice id="txadvice"> < !--配置事务参数--> < tx:attributes> < !--指定哪种规则的方法上面添加事务--> < tx:method name="accountMoney" propagation="REQUIRED"/> < !--< tx:method name="account*"/> --> < /tx:attributes> < /tx:advice> < !--3 配置切入点和切面--> < aop:config> < !--配置切入点--> < aop:pointcut id="pt" expression="execution(* com.atguigu.spring5.service.UserService.*(..))"/> < !--配置切面--> < aop:advisor advice-ref="txadvice" pointcut-ref="pt"/> < /aop:config>

5.7 事务操作(完全注解声明式事务管理)
  • 创建配置类,使用配置类替代xml配置文件
    @Configuration //配置类 @ComponentScan(basePackages = "com.atguigu") //组件扫描 @EnableTransactionManagement //开启事务 public class TxConfig //创建数据库连接池 @Bean public DruidDataSource getDruidDataSource() DruidDataSource dataSource = new DruidDataSource(); dataSource.setDriverClassName("com.mysql.jdbc.Driver"); dataSource.setUrl("jdbc:mysql:///user_db"); dataSource.setUsername("root"); dataSource.setPassword("root"); return dataSource; //创建JdbcTemplate对象 @Bean public JdbcTemplate getJdbcTemplate(DataSource dataSource) //到ioc容器中根据类型找到dataSource JdbcTemplate jdbcTemplate = new JdbcTemplate(); //注入dataSource jdbcTemplate.setDataSource(dataSource); return jdbcTemplate; //创建事务管理器 @Bean public DataSourceTransactionManager getDataSourceTransactionManager(DataSource dataSource) DataSourceTransactionManager transactionManager = new DataSourceTransactionManager(); transactionManager.setDataSource(dataSource); return transactionManager;

最后::: hljs-center
新人制作,如有错误,欢迎指出,感激不尽!
:::
::: hljs-center
欢迎关注公众号,会分享一些更日常的东西!
:::
::: hljs-center
如需转载,请标注出处!
:::
::: hljs-center
Spring | Spring5学习笔记#yyds干货盘点#

文章图片

:::

    推荐阅读