您现在的位置是:首页 >技术杂谈 >27事务管理&AOP网站首页技术杂谈
27事务管理&AOP
简介27事务管理&AOP
一、MySQL事务回顾
data:image/s3,"s3://crabby-images/c30ec/c30ec0525e4a800de1b21f8c5fdcbbbc1aebd4b1" alt="image-20230519095555441"
二、Spring事务管理
Spring框架的第一大核心:IOC控制反转
data:image/s3,"s3://crabby-images/b9283/b9283c76d198dbd45879350371ed154901cc1ae9" alt="image-20230519093354025"
data:image/s3,"s3://crabby-images/b044d/b044d30cbf2fb262ec8518f8783bb86d3c9f52a1" alt="image-20230519095638521"
在DeptServiceImpl下删除部门方法下新加一个删除员工信息的操作,注意:此时的id是部门id。
data:image/s3,"s3://crabby-images/6ad10/6ad10cde5090768d32f9b684df4d1e358b93707a" alt="image-20230519094945355"
data:image/s3,"s3://crabby-images/c9396/c9396feb83c815dcabb4b46b80fc823b7285b5ba" alt="image-20230519095242773"
1、问题分析
data:image/s3,"s3://crabby-images/cdcdb/cdcdb9b31faf4512aca8bd72c4c404b6c4565f09" alt="image-20230519095758677"
2、@Transactional-Spring事务管理
data:image/s3,"s3://crabby-images/ad5a6/ad5a64030123a703c211c5d1e9cde393ce5a5f4e" alt="image-20230519100017409"
一般是在Service实现类的方法上加Transactional注解
执行多次数据访问的增删改操作上需要用到事务
data:image/s3,"s3://crabby-images/2365b/2365b67e705c645f456ae440274f298b4851610a" alt="image-20230519101345091"
data:image/s3,"s3://crabby-images/a8722/a8722016e23cffcdac15161d31b735c9920196fc" alt="image-20230519101246113"
data:image/s3,"s3://crabby-images/5d725/5d7251f95c44c71b987e2f027761fa0ce26a1b51" alt="image-20230519101218784"
data:image/s3,"s3://crabby-images/ba231/ba231973f7c2ea6d162b66f8d929794997f2a664" alt="image-20230519101511274"
data:image/s3,"s3://crabby-images/e236e/e236e244681a568296559fd4aa82ccdcd1433a2e" alt="image-20230519102942049"
三、Spring事务进阶
data:image/s3,"s3://crabby-images/52952/529528b4706412f307e8a2f9ed141fa90db06581" alt="image-20230519103046587"
1、rollbackFor
data:image/s3,"s3://crabby-images/759f9/759f9d9660df14955e8547bdc3273afb334323fe" alt="image-20230519103807812"
Int i = 1/0 是属于运行时异常
data:image/s3,"s3://crabby-images/9ff24/9ff24be98352048bd53417bf744c73382331b962" alt="image-20230519104208589"
data:image/s3,"s3://crabby-images/307b2/307b27e20ebd5d352d29e5f95db451b92742c77e" alt="image-20230519110308309"
这是service层,需要继续往上抛异常
data:image/s3,"s3://crabby-images/78d0c/78d0ce393407bdcb339786ec3da0d34a9b055c74" alt="image-20230519110226786"
data:image/s3,"s3://crabby-images/6b768/6b768723cf8748cb227c7cf1c917c3d0531b470e" alt="image-20230519110436516"
默认情况下只有运行时异常才会进行事务回滚
2、propagation
data:image/s3,"s3://crabby-images/77b54/77b54d91cd2a63daf5cd53be8d661df3647505c1" alt="image-20230519133216508"
data:image/s3,"s3://crabby-images/7223d/7223d0bbff4204cdbe08a7da65a284637f24f3fe" alt="image-20230519133416156"
data:image/s3,"s3://crabby-images/50b62/50b629a82e999db407aee01f0a6590fe83ab021b" alt="image-20230519133437828"
data:image/s3,"s3://crabby-images/7ada6/7ada6560acdd14d13e978bf3d2e841372282cce7" alt="image-20230519133449823"
因为我们需要解散部门时,无论成功还是失败,都要记录操作日志,所以需要用到@Transaction的REQUIRES_NEW的属性,来新建一个事务
data:image/s3,"s3://crabby-images/f4351/f43510127929e35bda9e3d2c4442ea800d8be625" alt="image-20230519133808397"
pojo/DeptLog
package com.itheima.pojo;
import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;
import java.time.LocalDateTime;
@Data
@NoArgsConstructor
@AllArgsConstructor
public class DeptLog {
private Integer id;
private LocalDateTime createTime;
private String description;
}
DeptServiceImple
package com.itheima.service.impl;
import com.itheima.mapper.DeptLogMapper;
import com.itheima.mapper.DeptMapper;
import com.itheima.mapper.EmpMapper;
import com.itheima.pojo.Dept;
import com.itheima.pojo.DeptLog;
import com.itheima.service.DeptLogService;
import com.itheima.service.DeptService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional;
import java.time.LocalDateTime;
import java.util.List;
import java.util.logging.LogManager;
@Service
public class DeptServiceImpl implements DeptService {
@Autowired
private DeptMapper deptMapper;
@Autowired
private EmpMapper empMapper;// 删除部门id的同时,需要删除与部门id关联的员工信息
@Autowired
private DeptLogService deptLogService;// 添加日志对象注入
// @Autowired
// private DeptLogMapper deptLogMapper;
/*1、查询操作* */
// 实现方法
public List<Dept> list(){
List<Dept> deptList = deptMapper.select();
return deptList;
}
/*2、删除指定id
* */
@Transactional(rollbackFor = Exception.class) // 交给了Spring进行事务管理,将所有异常进行事务处理
@Override
public void deleteById(Integer id) throws Exception{
try {
deptMapper.deleteById(id); // 删除部门id
int i = 1/0;
// if (true){throw new Exception("出错了...");}
empMapper.deleteById(id); // 删除员工信息
} finally {// 记录日志描述
DeptLog deptLog = new DeptLog();
deptLog.setCreateTime(LocalDateTime.now());
deptLog.setDescription("执行了解散部门的操作,此次解散的部门id是"+id);
deptLogService.insert(deptLog);
/**
*可以不用写DeptLogService和DeptServiceImpl,直接写一个DeptLogMapper然后交给IOC控制器,再在
* 这个实现类中注入DeptLogMapper的对象直接调用insert方法,在方法前加入
* @Transactional(propagation = Propagation.REQUIRES_NEW)
*/
// deptLogMapper.insert(deptLog);
}
}
/*3、新增部门*/
public void add(Dept dept){
dept.setCreateTime(LocalDateTime.now());
dept.setUpdateTime(LocalDateTime.now());
deptMapper.insert(dept);
}
/*根据ID查询*/
public Dept selectById(Integer id){
Dept dept = deptMapper.selectById(id);
return dept;
}
@Override
/*更新部门名称*/
public void update(Dept dept) {
dept.setUpdateTime(LocalDateTime.now());
dept.setCreateTime(LocalDateTime.now());
deptMapper.update(dept);
}
}
DpetLogService
package com.itheima.service;
import com.itheima.pojo.DeptLog;
public interface DeptLogService {
void insert(DeptLog deptLog);
}
DeptLogServiceImpl
package com.itheima.service.impl;
import com.itheima.mapper.DeptLogMapper;
import com.itheima.pojo.DeptLog;
import com.itheima.service.DeptLogService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Propagation;
import org.springframework.transaction.annotation.Transactional;
@Service
public class DeptLogServiceImpl implements DeptLogService {
@Autowired
private DeptLogMapper deptLogMapper;
// 开启一个新事务
@Transactional(propagation = Propagation.REQUIRES_NEW)
@Override
public void insert(DeptLog deptLog) {
deptLogMapper.insert(deptLog);
}
}
DeptLogMapper
package com.itheima.mapper;
import com.itheima.pojo.DeptLog;
import org.apache.ibatis.annotations.Insert;
import org.apache.ibatis.annotations.Mapper;
import org.springframework.transaction.annotation.Propagation;
import org.springframework.transaction.annotation.Transactional;
@Mapper
public interface DeptLogMapper {
@Insert("insert into dept_log(create_time, description) VALUES (#{createTime},#{description})")
// @Transactional(propagation = Propagation.REQUIRES_NEW)
public void insert(DeptLog deptLog);
}
四、AOP基础-Spring框架的第二大核心
AOP概述
data:image/s3,"s3://crabby-images/75e2b/75e2be6f03877d96ea23ee9c60c0f59fc96be622" alt="image-20230519144922918"
data:image/s3,"s3://crabby-images/f06a9/f06a90b00f1a30f4d0e013ab82a0d21fc3c6fcb2" alt="image-20230519144936888"
AOP快速入门
data:image/s3,"s3://crabby-images/afeb7/afeb74cfd3a74ae969550b8fc6f45c8344e3097a" alt="image-20230519160123162"
package com.itheima.aop;
import lombok.extern.slf4j.Slf4j;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.springframework.stereotype.Component;
@Slf4j
@Component // 交给IOC容器
@Aspect // 表示这个是AOP类
public class TimeAspect {
// 当前共性功能应该加在哪些方法上
@Around("execution(* com.itheima.service.*.*(..))") //切入点表达式
public void recordTime(ProceedingJoinPoint joinPoint) throws Throwable {
// 1、获取方法运行开始时间
long begin = System.currentTimeMillis();
// 2、运行原始方法,o是原始方法的返回值
Object o = joinPoint.proceed();
// 3、获取方法结束运行结束时间
long end = System.currentTimeMillis();
// 计算方法耗时
// joinPoint.getSignature():原始方法的签名
log.info(joinPoint.getSignature()+"方法耗时:{}ms",end-begin);
}
}
data:image/s3,"s3://crabby-images/6f215/6f2153939fdb8f2296517f1c6de08a4589fc67cd" alt="image-20230519160259797"
data:image/s3,"s3://crabby-images/f9cbe/f9cbe4a283bc24ee15f4e7c25d0770de28d3bc54" alt="image-20230519161148430"
小结
data:image/s3,"s3://crabby-images/01f8c/01f8c57d32e243dcdbb0aa5e5059c9eb6ae888c8" alt="image-20230519161318405"
五、AOP基础-核心概念
AOP核心概念
data:image/s3,"s3://crabby-images/b7839/b7839a338e7d326479337b9a9688c1cb377c0e66" alt="image-20230519164857827"
AOP执行流程
data:image/s3,"s3://crabby-images/bff9d/bff9d87291e101ec9f1e284c7386efaaf26b072f" alt="image-20230519164715406"
小结
连接点JoinPoint | 能够被AOP所控制的方法 |
---|---|
切入点PointCut | 实际被AOP控制的方法,通过切入点表达式 |
通知Advice | 将所有共性功能封装起来的方法 |
切面Aspect | 描述通知与切入点之间的对应关系 |
目标对象Target | 通知所对应的对象 |
data:image/s3,"s3://crabby-images/67767/67767d19fc45f76f6d5151741f2142f0a7d0b34f" alt="image-20230519170101230"
六、AOP进阶
data:image/s3,"s3://crabby-images/4c080/4c08012944a367af77032e9edea45ec0d6941248" alt="image-20230519171259826"
1、通知类型
data:image/s3,"s3://crabby-images/a2101/a210109cd4c9c59446653d2b5e6f52bbb84e82c6" alt="image-20230519174120073"
data:image/s3,"s3://crabby-images/7302b/7302bb10c21629f9bb1e29b193e45c3cb3e1a2d1" alt="image-20230519174551031"
切入点表达式抽取
data:image/s3,"s3://crabby-images/5914b/5914b1900a0ef49b04f5a4407ca7081c4c516ac4" alt="image-20230519174957352"
data:image/s3,"s3://crabby-images/9735b/9735bda2ebc165d89aa48da323379d738b802c34" alt="image-20230519175305641"
小结
data:image/s3,"s3://crabby-images/c50b7/c50b78e3984c9ddd17e426c03b3c50b9f8bb0036" alt="image-20230519175400903"
2、通知顺序
data:image/s3,"s3://crabby-images/c55f4/c55f4d5c2de6277046b9c96a06433ad1f4e2bb2e" alt="image-20230523105728939"
data:image/s3,"s3://crabby-images/36f2d/36f2dced78d57c32b31e8f0f750a1b88f63aa1fe" alt="image-20230523110339906"
3、切入点表达式
execution
data:image/s3,"s3://crabby-images/13a04/13a04314e436076413ed45731bbb2155b67fdede" alt="image-20230523110701548"
data:image/s3,"s3://crabby-images/e60a7/e60a71dfcc17b72f76177dd043f69772173c242a" alt="image-20230523113303585"
data:image/s3,"s3://crabby-images/66fdc/66fdc152b0323d6f5fb5646e3366525bff881fc9" alt="image-20230523151557536"
data:image/s3,"s3://crabby-images/371ca/371ca000d387cec5cfc9347cb08864b62fd4e131" alt="image-20230523151743153"
MyAspect6.java
package com.itheima.aop;
import lombok.extern.slf4j.Slf4j;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;
import org.springframework.stereotype.Component;
@Component
@Slf4j
@Aspect
public class MyAspect6 {
// @Pointcut("execution(public void com.itheima.service.impl.DeptServiceImpl.delete(java.lang.Integer))")
// @Pointcut("execution(void com.itheima.service.DeptService.delete(java.lang.Integer))")//基于接口描述
// @Pointcut("execution(void delete(java.lang.Integer))")包名.类名不建议省略,匹配的范围过大,影响匹配的效率
// @Pointcut("execution(* com.itheima.service.DeptService.*(*))")//匹配返回值为任意,DeptService接口下所有方法任意类型的一个参数
// @Pointcut("execution(* com..DeptService.get(*))")//匹配返回值为任意,com包下任意子包中DeptService接口/类下get方法,为任意类型的一个参数
@Pointcut("execution(* com.itheima.service.DeptService.delete(java.lang.Integer)) ||"+
"execution(* com.itheima.service.DeptService.list())")
//匹配前面一个或者后面任意一个
public void pt(){}
@Before("pt()")
public void before(){
System.out.println("before ...6");
}
}
@annotation
data:image/s3,"s3://crabby-images/37b25/37b25324890aef464dbd65ea9397ede1d34c6e6d" alt="image-20230523153020619"
MyLog
package com.itheima.aop;
import lombok.extern.slf4j.Slf4j;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
@Retention(RetentionPolicy.RUNTIME)//该注解表示注解的生命周期
@Target(ElementType.METHOD)//表示该注解作用在哪一部分
public @interface MyLog {
}
MyAspect7
package com.itheima.aop;
import lombok.extern.slf4j.Slf4j;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;
import org.springframework.stereotype.Component;
@Slf4j
@Component
@Aspect
public class MyAspect7 {
@Pointcut("@annotation(com.itheima.aop.MyLog)")
public void pc(){}
@Before("pc()")
public void func(){
log.info("MyAspect7...");
}
}
data:image/s3,"s3://crabby-images/98d38/98d381a38ef27b251f0a36a4eb7eb67c687e9de1" alt="image-20230523160655953"
只需要在想要匹配的方法上加@MyLog注解就可以调用通知方法
小结
data:image/s3,"s3://crabby-images/e957e/e957e2b1d1f1e71c48bc9ab4d4b30623a37e8bcf" alt="image-20230523161211924"
4、连接点
data:image/s3,"s3://crabby-images/14af4/14af4ac3436f2e720503d8f61a34ff599cf383ac" alt="image-20230523162227036"
MyAspect8
package com.itheima.aop;
import lombok.extern.slf4j.Slf4j;
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.Signature;
import org.aspectj.lang.annotation.*;
import org.springframework.stereotype.Component;
import java.util.Arrays;
@Component
@Slf4j
@Aspect
public class MyAspect8 {
@Pointcut("execution(* com.itheima.service.DeptService.*(..))")
public void pt(){}
@Before("pt()")
public void before(JoinPoint joinPoint){
log.info("before....");
}
@Around("pt()")
public Object around(ProceedingJoinPoint joinPoint) throws Throwable {
log.info("MyAspect8 around before...");
// 1、获取目标类名
String className = joinPoint.getTarget().getClass().getName();
log.info("目标类名:{}",className);
// 2、获取目标方法签名
Signature signature = joinPoint.getSignature();
log.info("目标方法签名:{}",signature);
// 3、获取目标方法名
String signatureName = joinPoint.getSignature().getName();
log.info("目标方法名:{}",signatureName);
// 4、获取目标方法运行参数
Object[] args = joinPoint.getArgs();
String arg = Arrays.toString(args);
log.info("目标方法参数:{}",arg);
// 5、执行原始方法,获取返回值
Object result = joinPoint.proceed();
log.info("目标方法的返回值:{}",result);
log.info("MyAspect8 after ...");
return result;
}
@After("pt()")
public void after(){
log.info("after...");
}
}
测试类
package com.itheima;
import com.itheima.service.DeptService;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
@SpringBootTest
class SpringbootAopQuickstart1ApplicationTests {
@Autowired
DeptService deptService;
@Test
public void test1(){
deptService.delete(10);
}
@Test
public void test2(){
deptService.list();
}
}
执行delete之后
data:image/s3,"s3://crabby-images/de15e/de15eeb521c8d8f4654929aac9ba4b11a7c6e161" alt="image-20230523165601716"
只有环绕通知才可以执行原始方法
前置通知在原始方法执行前运行,后置通知在原始方法执行后运行
七、AOP案例
data:image/s3,"s3://crabby-images/36b88/36b881c128d344b2f014d887f926888fcd15847a" alt="image-20230523171728414"
1、准备工作
<!--AOP依赖-->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-aop</artifactId>
<version>2.7.1</version>
</dependency>
-- AOP案例
-- 操作日志表
create table operate_log(
id int unsigned primary key auto_increment comment 'ID',
operate_user int unsigned comment '操作人ID',
operate_time datetime comment '操作时间',
class_name varchar(100) comment '操作的类名',
method_name varchar(100) comment '操作的方法名',
method_params varchar(1000) comment '方法参数',
return_value varchar(2000) comment '返回值',
cost_time bigint comment '方法执行耗时, 单位:ms'
) comment '操作日志表';
sql表格对应的实体类
package com.itheima.pojo;
import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;
import java.time.LocalDateTime;
@Data
@NoArgsConstructor
@AllArgsConstructor
public class OperateLog {
private Integer id; //ID
private Integer operateUser; //操作人ID
private LocalDateTime operateTime; //操作时间
private String className; //操作类名
private String methodName; //操作方法名
private String methodParams; //操作方法参数
private String returnValue; //操作方法返回值
private Long costTime; //操作耗时
}
2、编码
data:image/s3,"s3://crabby-images/eb3a6/eb3a621821cfd98423b7b969f27d3ac7909f55ad" alt="image-20230524154418945"
LogAspect.java
package com.itheima.aop;
import com.alibaba.fastjson.JSONObject;
import com.itheima.mapper.OperateLogMapper;
import com.itheima.pojo.OperateLog;
import com.itheima.utils.JwtUtils;
import io.jsonwebtoken.Claims;
import lombok.extern.slf4j.Slf4j;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;
import javax.servlet.http.HttpServletRequest;
import java.time.LocalDateTime;
import java.util.Arrays;
@Component
@Slf4j
@Aspect //切面类
public class LogAspect {
@Autowired//用于获取jwt令牌
private HttpServletRequest httpServletRequest;
@Autowired
private OperateLogMapper operateLogMapper;
@Around("@annotation(com.itheima.anno.Log)")//切点表达式
public Object recordLog(ProceedingJoinPoint joinPoint) throws Throwable {
// 1、获取操作人ID,当前登录员工id,这一步需要用到jwt令牌,在Header中
String jwt = httpServletRequest.getHeader("token");
// 解析jwt令牌
Claims claims = JwtUtils.parseJwt(jwt);
Integer operateUser = (Integer) claims.get("id");
// 2、操作时间
LocalDateTime operateTime = LocalDateTime.now();
// 3、操作类名
String className = joinPoint.getTarget().getClass().getName();
// 4、操作方法名
String methodName = joinPoint.getSignature().getName();
// 5、操作方法参数
Object[] args = joinPoint.getArgs();
String methodParams = Arrays.toString(args);
// 获取方法开始时间
long begin = System.currentTimeMillis();
// 6、操作方法返回值,转换为json格式的字符串保存起来
Object result = joinPoint.proceed();
String returnValue = JSONObject.toJSONString(result);
// 获取结束时间
long end = System.currentTimeMillis();
// 7、操作耗时
long costTime = end - begin;
// 将日志内容插入到表中
OperateLog operateLog = new OperateLog(null,operateUser,operateTime,className,methodName,methodParams,returnValue,costTime);
operateLogMapper.insert(operateLog);
log.info("AOP操作日志:{}",operateLog);
return result;//返回的是执行方法的返回值
}
}
OperateLogMapper
package com.itheima.mapper;
import com.itheima.pojo.OperateLog;
import org.apache.ibatis.annotations.Insert;
import org.apache.ibatis.annotations.Mapper;
@Mapper
public interface OperateLogMapper {
@Insert("insert into operate_log (operate_user, operate_time, class_name, method_name, method_params, return_value, cost_time) " +
"values (#{operateUser}, #{operateTime}, #{className}, #{methodName}, #{methodParams}, #{returnValue}, #{costTime});")
public void insert(OperateLog operateLog);
}
注意
data:image/s3,"s3://crabby-images/95341/95341bba0bd29852e040ac0e395fd5f2be95c5a5" alt="image-20230524155552227"
风语者!平时喜欢研究各种技术,目前在从事后端开发工作,热爱生活、热爱工作。