风流不在谈锋胜,袖手无言味最长。这篇文章主要讲述Spring | Spring5学习笔记#yyds干货盘点#相关的知识,希望能为你提供帮助。
Spring5学习笔记
0 总结:
- Spring框架概述
(1)轻量级开源javaEE框架,为了解决企业复杂性,两个核心组成:IOC和AOP
(2)Spring5.2.6版本
- IOC容器
(1)IOC底层原理(工厂、反射等)
(2)IOC接口(BeanFactory)
(3)IOC操作Bean管理(基于xml)
(4)IOC操作Bean管理(基于注解)
- Aop
(1)AOP底层原理:动态代理,有接口(JDK动态代理),没有接口(CGLIB动态代理)
(2)术语:切入点、增强(通知)、切面
(3)基于AspectJ实现AOP操作
- JdbcTemplate
(1)使用JdbcTemplate实现数据库curd操作
(2)使用JdbcTemplate实现数据库批量操作
- 事务管理
(1)事务概念
(2)重要概念(传播行为和隔离级别)
(3)基于注解实现声明式事务管理
(4)完全注解方式实现声明式事务管理
- Spring是轻量级的开源的JavaEE框架
- Spring可以解决企业应用开发的复杂性
- Spring有两个核心部分:IOC和Aop
(1)IOC:控制反转,把创建对象过程交给Spring进行管理
(2)Aop:面向切面,不修改源代码进行功能增强
- Spring特点
(1)方便解耦,简化开发
(2)Aop编程支持
(3)方便程序测试
(4)方便和其他框架进行整合
(5)方便进行事务操作
(6)降低API开发难度
- 什么是IOC
(1)控制反转,把对象创建和对象之间的调用过程,交给Spring进行管理
(2)使用IOC目的:为了耦合度降低
(3)做入门案例就是IOC实现
- IOC底层原理
(1)xml解析、工厂模式、反射
- IOC思想基于IOC容器完成,IOC容器底层就是对象工厂
- 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);
- ApplicationContext接口有实现类
(1)FileSystemXmlApplicationContext
:
(2)ClassPathXmlApplicationContext
:
- 什么是Bean管理
- Bean管理操作有两种方式
?(2)基于注解方式实现
2.3 IOC操作Bean管理(基于xml方式)
- 基于xml方式创建对象
< !--1 配置User对象创建--> < bean id="user" class="com.atguigu.spring5.User"> < /bean>
(1)在spring配置文件中,使用bean
标签,标签里面添加对应属性,就可以实现对象创建
(2)在bean标签有很多属性,常用的属性
> **`id`**属性:唯一标识
(3)创建对象时候,默认也是执行 < u> 无参数构造< /u> 方法完成对象创建
- 基于xml方式注入属性
(1)DI:依赖注入,就是注入属性
(2)DI是IOC的一种具体实现,表示依赖注入,或者说注入属性,但需要在创建对象的基础之上完成
- 第一种注入方式:使用
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>
- 第二种注入方式:使用有参数构造进行注入
(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>
- 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>
- 字面量
(1)null值
< !--null值--> < property name="address"> < null/> < /property>
(2)属性值包含特殊符号
- 把<
>
进行转义
& lt
;& gt
; - 把带特殊符号内容写到
CDATA
< property name="address"> < value> < ![CDATA[< < 南京> > ]]> < /value> < /property>
- 把<
>
进行转义
- 注入属性-外部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>
- 注入属性-内部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>
- 注入属性-级联赋值
(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>
- 注入数组类型属性
- 注入List集合类型属性
- 注入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>
- 在集合里面设置对象类型值
< !--注入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>
- 把集合注入部分提取出来
(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>
- 普通bean:在配置文件中定义bean类型就是返回类型
- 工厂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>
- 在Spring里面,设置创建bean实例是单实例还是多实例
- 在Spring里面,默认情况下,bean是单实例对象
- 如何设置单实例还是多实例
(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)singleton
和prototype
区别
singleton
单实例,prototype
多实例- 设置
scope
值是singleton
时候,加载spring
配置文件时候就会创建单实例对象 - 设置
scope
值是prototype
时候,不是在加载spring
配置文件时候创建对象,在调用getBean
方法时候创建多实例对象
- 第一个值默认值,
- 生命周期
(1)从对象创建到对象销毁的过程
- bean生命周期
(1)通过构造器创建bean
实例(无参数构造)
(2)为bean
的属性设置值和对其他bean
引用(调用set
方法)
(3)调用bean
的初始化的方法(需要进行配置初始化的方法)
(4)bean
可以使用了(对象获取到了)
(5)当容器关闭时候,调用bean
的销毁的方法(需要进行配置销毁的方法)
- 演示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();
- bean的后置处理器,bean生命周期有七步
(1)通过构造器创建bean
实例(无参数构造)
(2)为bean
的属性设置值和对其他bean
引用(调用set方法)
(3)把bean
实例传递bean后置处理器的方法postProcessBeforeInitialization
(4)调用bean
的初始化的方法(需要进行配置初始化的方法)
(5)把bean实例传递bean后置处理器的方法postProcessAfterInitialization
(6)bean
可以使用了(对象获取到了)
(7)当容器关闭时候,调用bean
的销毁的方法(需要进行配置销毁的方法)
- 演示添加后置处理器效果
(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>
- 自动装配:根据指定装配规则(属性名称或者属性类型),Spring自动将匹配的属性值进行注入
- 自动装配示例
(1)bean标签属性autowire
,配置自动装配
(2)autowire
属性常用两个值:
byName
根据属性名称注入 ,注入值bean
的id
值和类属性名称一样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>
- 直接配置数据库信息
(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>
- 引入外部属性文件配置数据库连接池(建议)
(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>
- 引入
- 什么是注解
(1)注解是代码特殊标记,格式:@注解名称(属性名称=属性值,属性名称=属性值..)
(2)使用注解,注解作用在类上面,方法上面,属性上面
(3)使用注解目的:简化xml配置
- Spring针对Bean管理中创建对象提供注解
(1)@Repository
:数据库层组件,DAO层;(2)**`@Service`**:业务逻辑组件Service层;
br/>(2)**`@Service`**:业务逻辑组件Service层;
@Controller:控制器,Controller层;
(4)@Component
:组件,POJO层。
- 基于注解方式实现对象创建
(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.......");
- 开启组件扫描细节配置
(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>
context:exclude-filter
表示设置不扫描哪些内容
< context:component-scan base-package="com.atguigu"> < context:exclude-filter type="annotation" expression="org.springframework.stereotype.Controller"/> < /context:component-scan>
- 当
- 基于注解方式实现属性注入
(1)@Autowired
:根据属性类型进行自动装配
- 第一步把
service
和dao
对象创建,在service
和dao
类添加创建对象注解
- 第二步在
service
注入dao
对象,在service
类添加dao
类型属性,在属性上面使用注解
- 类型相当于类名
@Service public class UserService //定义dao类型属性 //不需要添加set方法 //添加注入属性注解 @Autowired private UserDao userDao; public void add() System.out.println("service add......."); userDao.add();
@Qualifier
:根据名称进行注入(要和@Autowired
一起使用)
- 名称相当于对象名
@Autowired//根据类型进行注入 @Qualifier(value = "https://www.songbingjia.com/android/userDaoImpl1") //根据名称进行注入 private UserDao userDao;
@Resourse
:可以根据类型注入,也可以根据名称注入
- 相当于
@Autowired
与@Qualifier
//@Resource//根据类型进行注入 @Resource(name = "userDaoImpl1")//根据名称进行注入 private UserDao userDao;
@Value
:注入普通类型属性
- 普通注入
@Value(value = "https://www.songbingjia.com/android/abc") private String name;
- 第一步把
- 完全注解开发(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();
- 创建配置类,替代
(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动态代理底层原理)
- 使用JDK动态代理,使用Proxy类里面的方法创建代理对象
(1)调用newProxyInstance
方法
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-e8Itd0IA-1623595927681)(D:\\桌面文件夹\\校内文件\\大二下\\我的Markdowm笔记\\spring5图片\\newProxyInstance.png)]
方法有三个参数:
- 第一参数,类加载器;
- 第二参数,增强方法所在的类,这个类实现的接口,支持多个接口;
- 第三参数,实现这个接口
InvocationHandler
,创建代理对象,写增强的部分;
- 编写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;
- 连接点
- 类里面哪些方法可以被增强,这些方法称为连接点
- 切入点
- 实际被真正增强的方法
- 通知(增强)
- 实际增强的逻辑部分
- 通知有多种类型:
- 实际增强的逻辑部分
- 切面
- 是动词
- 把通知应用到切入点的过程
- Spring框架一般都是基于
AspectJ
实现AOP操作
- AspectJ不是Spring组成部分,独立AOP框架,一般把AspectJ和Spirng框架一起使用,进行AOP操作
- 基于AspectJ实现AOP操作
(1)基于xml配置文件实现
(2)基于注解方式实现(使用)
- 在项目工程里面引入AOP相关依赖
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-Ts8zfehP-1623595927683)(D:\\桌面文件夹\\校内文件\\大二下\\我的Markdowm笔记\\spring5图片\\AOP相关依赖.png)]
- 切入点表达式
(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.. (..))`**
- 举例1:对com.atguigu.dao.BookDao类里面的add进行增强
- 创建两个类,增强类和被增强类,创建方法
public class Book public void buy() System.out.println("buy.............");
public class BookProxy public void before() System.out.println("before.........");
- 在spring配置文件中创建两个类对象
< !--创建对象--> < bean id="book" class="com.atguigu.spring5.aopxml.Book"> < /bean> < bean id="bookProxy" class="com.atguigu.spring5.aopxml.BookProxy"> < /bean>
- 在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>
- 创建类,并在类里定义方法
public class User public void add() System.out.println("add.......");
- 创建增强类(编写增强逻辑)
//增强的类 public class UserProxy public void before()//前置通知 System.out.println("before.........");
- 进行通知的配置
(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)使用注解创建User
和UserProxy
对象
@Component public class User
@Component public class UserProxy
(3)在增强类上面添加注解@Aspect
@Component @Aspect //生成代理对象 public class UserProxy
(4)在spring配置文件中开启生成代理对象
< !-- 开启Aspect生成代理对象--> < aop:aspectj-autoproxy> < /aop:aspectj-autoproxy>
- 配置不同类型的通知
- 在增强类的里面,在作为通知方法上面添加通知类型注解,使用切入点表达式配置
- 可以进行相同的切入点抽取,使用
@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("环绕之后.........");
- 在增强类的里面,在作为通知方法上面添加通知类型注解,使用切入点表达式配置
- 有多个增强类同一个方法进行增强,设置增强类优先级
- 在增强类上面添加注解
@Order
(数字类型值),数字类型值越小优先级越高
@Component @Aspect @Order(1) public class PersonProxy
- 在增强类上面添加注解
- 完全使用注解开发
- 创建配置类,不需要创建xml配置文件
@Configuration @ComponentScan(basePackages = "com.atguigu") @EnableAspectJAutoProxy(proxyTargetClass = true) public class ConfigAop
- 创建配置类,不需要创建xml配置文件
- 什么是JdbcTemplate
(1)Spring框架对JDBC进行封装,使用JdbcTemplate
方便实现对数据库操作
- 准备工作
(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
<
!-- 数据库连接池 -->
<
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操作数据库(添加)
- 对应数据库创建实体类
- 编写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);
- 第一个参数:sql语句
//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操作数据库(查询返回某个值)
- 查询表里面有多少条记录,返回是某个值
- 使用JdbcTemplate实现查询返回某个值代码
?有两个参数
- 第一个参数: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操作数据库(查询返回对象)
- 场景:查询图书详情
- JdbcTemplate实现查询返回对象
?有三个参数
- 第一个参数: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操作数据库(查询返回集合)
- 场景:查询图书列表分页…
- 调用JdbcTemplate方法实现查询返回集合
有三个参数
- 第一个参数: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操作数据库(批量操作)
- 批量操作:操作表里面多条记录
- 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);
- 第一个参数:sql语句
- 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);
- 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);
- 什么事务
(1)事务是数据库操作最基本单元,逻辑上一组操作,要么都成功,如果有一个失败所有操作都失败
(2)典型场景:银行转账
- lucy转账100元给mary
- lucy少100,mary多100
- 事务四个特性(ACID)
(1)原子性
(2)一致性
(3)隔离性
(4)持久性
- 创建数据库表,添加记录
- 创建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;
- 在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();
- 事务添加到JavaEE三层结构里面Service层(业务逻辑层)
- 在Spring进行事务管理操作
(1)有两种方式:编程式事务管理和声明式事务管理(使用)
- 声明式事务管理
(1)基于注解方式(使用)
(2)基于xml配置文件方式
- 在Spring进行声明式事务管理,底层使用AOP原理
- Spring事务管理API
(1)提供一个接口,代表事务管理器,这个接口针对不同的框架提供不同的实现类
DataSourceTransactionManager(org.springframework.jdbc.datasource)
- 在spring配置文件配置事务管理器
< !--创建事务管理器--> < bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager"> < !--注入数据源--> < property name="dataSource" ref="dataSource"> < /property> < /bean>
- 在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>
- 在service类上面(或者service类里面方法上面)添加事务注解
(1)@Transactional
,这个注解添加到类上面,也可以添加方法上面
(2)如果把这个注解添加类上面,这个类里面所有的方法都添加事务
(3)如果把这个注解添加方法上面,为这个方法添加事务
@Service @Transactional public classUserService
- 在service类上面添加注解@Transactional,在这个注解里面可以配置事务相关参数
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-4PY8x1y5-1623595927690)(D:\\桌面文件夹\\校内文件\\大二下\\我的Markdowm笔记\\spring5图片\\Transactional注解.png)]
- 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
- 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
- timeout:超时时间
(1)事务需要在一定时间内进行提交,如果不提交进行回滚
(2)默认值是-1,设置时间以秒单位进行计算
- readOnly:是否只读
(1)读:查询操作,写:添加修改删除操作
(2)readOnly默认值false,表示可以查询,可以添加修改删除操作
(3)设置readOnly值是true,设置成true之后,只能查询
- rollbackFor:回滚
(1)设置出现哪些异常进行事务回滚
- noRollbackFor:不回滚
(1)设置出现哪些异常不进行事务回滚
- 在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>
- 创建配置类,使用配置类替代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
文章图片
:::
推荐阅读
- RENIX操作之XML报文模板说明——网络测试仪实操
- #yyds干货盘点# 30个类手写Spring核心原理之动态数据源切换
- #私藏项目实操分享# 你了解shiro吗(手把手教你集成shiro)
- Veeam Backup for Red Hat Virtualization
- 快速入门vue,含实战案例,“建议收藏”,用到省的百度了#yyds干货盘点#
- IDEA对Docker容器进行打包构建的两种方式
- 「自我检验」熬夜总结50个Vue知识点,全都会你就是神!!!
- MongoDB学习笔记-使用 MongoDB 进行 CRUD 操作(上)
- #yyds干货盘点#mybatis-plus学习与实践代码生成器整合swagger2生成CRUD接口