JavaWeb回顾与小结(七)

事务管理

事务回顾

概念

事务是一组操作的集合,它是一个不可分割的工作单位,这些操作要么同时成功,要么同时失败

操作

开启事务

start transaction/begin
一组操作开始前,开启事务

提交事务

commit
这组操作全部成功后,提交事务

回滚事务

rollback
中间任何一个操作出现异常,回滚事务

Spring事务管理

注解

@Transactional

位置

业务(service)层的方法上,类上,接口上

作用

将当前方法交给spring进行事务管理,方法执行前,开启事务;成功执行完毕,提交事务;出现异常,回滚事务

开启事务管理日志

logging:
  level:
    org.springframework.jdbc.support.JdbcTransactionManager:debug

事务进阶

回滚

rollback
默认情况下,只有出现RuntimeException才回滚异常.rollbFor属性用于控制出现何种异常类型,回滚事务

传播行为

propagation

事务传播行为:指的就是当一个事务方法被另一个事务方法调用时,这个事务方法应该如何进行事务控制

属性值

REQUIRED
(默认值)需要事务,有责加入,无则创建新事务
REQUIRES_NEW
需要新事务,无论有无,总是创建新事务
SUPPORTS
支持事务,有则加入,无则在无事务状态中运行
NOT_SUPPORTED
不支持事务,在无事务状态下运行,如果当前存在已有事务,则挂起当前事务
MANDATORY
必须有事务,否则抛异常
NEVER
必须没事务,否则抛异常
...

场景
REQUIRED

大部分情况下都是用该传播行为即可

REQUIRES_NEW

当我们不希望事务之间互相影响时,可以使用该传播行为.比如,下订单前需要记录日志,不论订单保存成功与否,都需要保证日志记录能够记录成功

AOP基础

AOP概述

AOP

Aspect Oriented Programming(面向切面编程,面向方面编程),其实就是面向特定方法编程

场景

案例部分功能运行较慢,定位执行耗时较长的业务方法,此时需要统计每一个业务方法的执行耗时

实现

动态代理是面向切面编程最主流的实现,而SpringAOP是Spring框架的高级技术,旨在管理bean对象的过程中,主要通过底层的动态代理机制,对特定的方法进行编程

AOP快速入门

步骤

  1. 导入依赖
  2. 编写AOP程序,针对于特定方法根据业务需要进行编程

AOP

场景
  • 记录操作日志
  • 权限控制
  • 事务管理
优势
  • 代码无侵入
  • 减少重复代码
  • 提高开发效率
  • 维护方便

AOP核心概念

连接点

JoinPoint,可以被AOP控制的方法(暗含方法执行时的相关信息)

通知

Advice,指哪些重复的逻辑,也就是共性功能(最终体现为一个方法)

切入点

Aspect,描述通知与切入点的对应关系(通知+切入点)

目标对象

Target,通知所应用的对象

AOP进阶

通知类型

@Around
环绕通知,此注解标注的通知方法在目标方法前,后都被执行
@Before
前置通知,此注解标注的通知方法在目标方法前被执行
@After
后置通知,此注解标注的通知方法在目标方法后被执行,无论是否有异常都会执行
@AfterReturning
返回后通知,此注解标注的通知方法在目标方法后被执行,有异常不会执行
@AfterThrowing
异常后通知,此注解标注的通知方法发生异常后执行

注意事项

  • @Around环绕通知需要自己调用ProceedingJoinPoint.proceed()来让原始方法执行,其他通知不需要考虑目标方法执行
  • @Around环绕通知方法的返回值,必须指定为Object,来接收原始方法的返回值

@PointCut
该注解的作用是将公共的切点表达式抽取出来,需要用到时引用该切点表达式即可

  • private:仅能在当前切面类中引用该表达式
  • public:在其他外部的切面类中也可以引用该表达式

通知顺序

当有多个切面的切入点都匹配到了目标方法,目标方法运行时,多个通知方法都会被执行

执行顺序

不同切面类中,默认按照切面类的类名字母排序
  • 目标方法前的通知方法:字母排名靠前的先执行
  • 目标方法后的通知方法:字母排名靠前的后执行
用@order(数字)加在切面类上来控制顺序
  • 目标方法前的通知方法:数字小的先执行
  • 目标方法后的通知方法:数字小的后执行

切入点表达式

描述切入点方法的一种表达式

作用

主要用来决定项目中的哪些方法需要加入通知

常见形式

  • execution(...):根据方法的签名来匹配
  • @annotation(...):根据注解匹配

切入点表达式-execution

主要根据方法的返回值,包名,类名,方法名,方法参数等信息来匹配,语法为:

execution(访问修饰符? 返回值 包名.类名.?方法名(方法参数) throws 异常?)

其中带?的表示可以省略的部分
  • 访问修饰符:可省略
  • 包名,类名:可省略
  • throws异常:可省略(注意是方法上声明抛出的异常,不是实际抛出的异常)
可以使用通配符描述切入点
  • *:单个独立的任意符号,可以通配任意返回值,包名,类名,方法名,任意类型的一个参数,也可以通配包,类,方法名的一部分

  • ..:多个连续的任意符号,可以通配任意层级的包,或任意类型,任意个数的参数

注意事项

根据业务需要,可以使用且(&&),或(||),非(!)来组合比较复杂的切入点表达式

书写建议
  • 所有业务方法名在命名时尽量规范,方便切入点表达式快速匹配,如:查询类方法都是find开头,更新类方法都是update开头
  • 描述切入点方法通常基于接口描述,而不是直接描述实现类,增强拓展性
  • 在满足业务需要的前提下,尽量缩小切入点的匹配范围,如:包名匹配尽量不使用"..",使用"*"来匹配单个包

切入点表达式-@annotation

用于匹配标识有特定注解的方法

连接点

在Spring中用JoinPoint抽象了连接点,用它可以获得方法执行时的相关信息,如目标类名,方法名,方法参数

  • 对于@Around通知,获取连接点信息只能使用ProceedingJoinPoint
  • 对于其他四种通知,获取连接点信息只能使用JoinPoint,它时ProceedingJoinPoint的父类型

AOP案例

步骤

准备

  • 在案例工程中引入AOP的起步依赖
  • 导入资料中准备好的数据库表结构,并引入对应的实体类

编码

  • 自定义注解@Log
  • 定义切面类,完成记录操作日志的逻辑
    获取request对象,从请求头中获取到jwt令牌,解析令牌获取当前用户的id

配置优先级

配置支持三种格式

properties
yml
yaml

注意事项

虽然springboot支持多种格式配置文件,但在项目开发时,推荐统一使用一种格式的配置(yml是主流)

SpringBoot除了支持配置文件属性配置,还支持Java系统属性和命令行参数的方式进行属性配置

  • Java系统属性
    -Dserver.port=9000

  • 命令行参数
    --server.port=10010

cmd命令操作方式

  • 执行maven打包指令package
    springboot项目打包时,需引入插件spring-boot-maven-plugin
  • 执行Java指令,运行jar包

Bean管理

获取bean

默认情况下,项目启动时,会把bean都创建好放在IOC容器中,如果想获取这些bean,通过如下方式:
根据name获取bean
Object getBean(String name)
根据类型获取bean
<T> T getBean(Class<T> requiredType)
根据name获取bean(带类型转换)
<T> T getBean(String name,Class<T> requiredType)
上述所说的项目启动时,bean会创建好,还会受到作用域及延迟初始化影响,这里主要针对于默认的单例非延迟加载的bean而言

bean作用域

spring支持五种作用域,后三种在web环境才生效

singleton
容器内同名称的bean只有一个实例(单例)(默认)
prototype
每次使用该bean时会创建新的实例(非单例)
request
每个请求范围内会创建新的实例(web环境中,了解)
session
每个会话范围内会创建新的实例(web环境中,了解)
application
每个应用范围内会创建新的实例(web环境中,了解)

可以通过@Scope注解来进行配置作用域

  • 默认singleton的bean,在容器启动时被创建,可以使用@Lazy注解来延迟初始化(延迟到第一次使用时)
  • prototype的bean,每一次使用该bean的时候都会创建一个新的实例
  • 实际开发当中,绝大部分的bean是单例的,也就是说绝大部分bean不需要配置scope属性

第三方bean

@Bean

  • 如果要管理的bean对象来自第三方(不是自定义的),是无法用@Component等注解声明bean的,就需要用到@Bean注解
  • 若要管理的第三方bean对象,建议对这些bean进行集中分类配置,可以通过@Configuration注解声明一个配置类

格式示例

@Configuration
public class CommonConfig{
  @Bean
  public SAXReader saxReander(){
    return new SAXReader();
  }
}

注意

  • 通过@Bean注解的name或value属性可以声明bean的名称,如果不指定,默认bean的名称就是方法名
  • 如果第三方bean需要依赖其他bean对象,直接在bean定义方法中设置形参即可,容器会根据类型自动装配

使用场景

  • 项目中自定义的bean交给spring管理,使用@Component及其衍生物注解
  • 项目中第三方的bean交给spring管理,使用@Bean注解

SpringBoot原理

起步依赖

原理是,利用了maven依赖传递的特性,把我们常用的依赖通过starter依赖间接传递进来

自动配置原理

概述

spring boot的自动配置就是当spring容器启动后,一些配置类,bean对象就自动存入到IOC容器中,不需要我们手动去声明,从而简化了开发,省去了繁琐的配置操作

方案

@ComponentScan组件扫描
@Import导入,使用import导入的类会被spring加载到IOC容器中,导入形式以下
  • 导入普通类
  • 导入配置类
  • 导入ImportSelector接口实现类(导入器)
    并不是程序一启动就加载所有配置类,通过@CondeitonalOnXxx注解定义条件,加载符号条件的配置类

条件注解@Conditional

作用

按照一定的条件进行判断,在满足给定条件后,才会创建对应的bean对象到spring IOC容器中

位置

方法,类

@Conditional本身是一个父注解,派生出大量的子注解

@ConditionalOnClass
判断环境中是否有对应字节码文件,才创建bean到IOC容器
@ConditionalOnMissingBean
判断环境中没有对应的bean(类型或名称),才创建bean到IOC容器
@ConditionalOnProperty
判断配置文件中有对应属性和值,才创建bean到IOC容器

源码分析

@SpringBootApplication

该注解标识在SpringBoot工程引导类上,是SPringleBoot中最最最重要的注解,该注解由三个部分组成
@SpringBootConfiguration
该注解与@Configuration注解作用相同,用来声明当前也是一个配置类
@ComponentScan
组件扫描,默认扫描当前引导类所在包及其子包
@EnableAutoConfiguration
SpringBoot实现自动化配置的核心注解

springboot怎么加载到自动配置包中的第三方bean

  • 每一个SpringBoot程序启动类上都有一个@SpringBootApplication注解,该注解内部使用@EnableAutoConfiguration注解开启自动配置

  • 在@EnableAutoConfiguration内部使用@Import注解通过ImportSelector导入批量加载自动配置包中的配置类

  • 具体加载过程是导入器会读取自动配置包中META-INF/spring.factories文件,获取文件中定义的配置类的全类名,判断类或者方法上的条件是否满足,如果满足就创建对应的bean对象保存到IOC容器中

自定义starter起步依赖

场景

在实际开发中,经常会定义一些公共组件,提供给各个项目团队使用,而在springboot项目中,一般会将这些公共组件封装为springboot的starter起步依赖

分析

需求

自定义aliyun-oss-spring-boot-starter,完成阿里云oss操作工具类AliyunOSSUtils的自动配置

目标

引入起步依赖引入后,要想使用阿里云OSS,注入AliyunOSSUtils直接使用即可

posted @ 2023-05-10 11:23  zFlame_5020  阅读(46)  评论(0)    收藏  举报