Spring BOOT AOP基础应用教程
作者:上后左爱 发布时间:2023-05-29 20:33:32
面试课题 Spring boot AOP
Spring boot 中 AOP是其中 重要的特性,其实现的方式借助的 * + Proxy * ,在AOP主要用于日志打印,安全拦截,事务处理,异常处理和性能统计,要向深刻了解Spring boot AOP 原理,从 Spring * 的原理讲起
Spring boot *
原理:
* 底层实现借助 java.lang.reflect.Proxy 的 newProxyInstance的方法
其有是三个参数:
1.Class的类加载器
2.接口方法
3.h 增强方式
在代码中 定于 interface , interfaceImpl 具体的实现类 ,使用 java 代理代码方式进行处理:
Proxy.newProxyInstance(Main.Class.getClassLoader(), new Class[] {UserDao.class}, new InvocationHandler() {
})
//在 InvocationHandler() 调用方法之前增强添加预处理 和 方法调用后的处理东西
public interface UserDao {
public int add(int a, int b);
}
public class UserDaoImpl implements UserDao {
@Override
public int add(int a, int b) {
System.out.println("add 方法执行了");
return a+b;
}
}
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import java.util.Arrays;
public class Main {
public static void main(String[] args) {
Class[] interfaces = {UserDao.class};
UserDaoImpl userDaoImpl = new UserDaoImpl();
//创建接口实现类代理对象
//此处用UserDao作为返回值的类型,是因为我们传入的interfaces就是UserDao.class
UserDao dao = (UserDao) Proxy.newProxyInstance(Main.class.getClassLoader(), interfaces, new InvocationHandler() {
//把想要代理的对象传递进来
private Object object = userDaoImpl;
//增强的逻辑
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
//方法之前
System.out.println("方法之前执行 : " + method.getName() + "; 传递的参数:" + Arrays.toString(args) + "; object:" + object);
//被增强的方法执行,填写要增强的对象、参数
Object res = method.invoke(object, args);
//方法之后
System.out.println("方法之后执行 : " + method.getName() + "; 传递的参数:" + Arrays.toString(args) + "; object:" + object);
return res;
}
});
int res = dao.add(1, 2);
System.out.println("这个是res: " + res);
}
}
总结
Spring boot中 能够实现AOP的底层原理,之上的代码属于静态编码方式 ,需要相同的逻辑抽象出来,因此诞生了AOP,在Spring boot中 * 有两种
基于接口的JDK- * (返回类型属于接口类型);
基于父类的cglib 代理,通过继承关系代理(不管是接口还是实现类 OK)
在实际使用过程中 Spring boot 默认 cglib * ,使用范围更加广泛
AOP 切面
基本知识
pointcut: 切入点: execution… 表示需要在哪些方法上生效,对哪些方法进行增强 – 使用正则表达式
Advice: 通知: 自定义处理 ,通知 分为BeforAdvice, AfterAdvice, ThrowAdvice
Advisor: 将 PointCut 与 Advice 进行连接起来定义哪些通知在哪些方法增强生效 – 对切面XXAOP 使用@Ascpect 注解进行生效定义
@Component
@Aspect
public class BookAop {
// 定义切入点
public static final String POINT_CUT = "execution(* com.example.bootaop.dao..*.*(..))";
@Before(POINT_CUT)
public void before() {
System.out.println("----------添加图书方法前[校验]-----------");
}
@After(POINT_CUT)
public void after(JoinPoint jp) {
System.out.println("----------添加图书成功后-----------");
System.out.println(jp.getTarget().getClass());
System.out.println(Arrays.asList(jp.getArgs()));
}
}
自定义注解
如上显示是 AOP的切面,但是AOP切面有个使用不好定法在于 pointcut 写正则表达式 无法准确的表达,最好有个插拔式方式 ,引入到自定义注解,自定义注解弥补这一缺陷
元注解
元注解是 java 自带的类型
@Retention 注解 保留策略(SOURCE,CLASS,RUNTIME)
@Retention(RetentionPolicy.SOURCE) 仅存在于源码中
@Retention(RetentionPolicy.CLASS) 存在于class字节码中,但运行时无法获取
@Retention(RetentionPolicy.RUNTIME) 存在于class字节码中,运行时可以通过反射获取
Target 注解 作用范围
@Target(ElementType.TYPE) 接口、类等
@Target(ElementType.FIELD) 字段
@Target(ElementType.METHOD) 方法
@Target(ElementType.PARAMETER) 方法参数
@Target(ElementType.CONSTRUCTOR) 构造函数
@Target(ElementType.LOCAL_VARIABLE) 局部变量
@Target(ElementType.ANNOTATION_TYPE) 注解
@Target(ElementType.PACKAGE) 包
自定义注解
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface MyLog {
String value() default "";
}
如何在AOP引用
在 pointcut = “@annotation(MyLog)”
@Slf4j
@Aspect
@Component
public class LogAspect {
@Around("@annotation(myLog)")
public Object around(ProceedingJoinPoint point, MyLog myLog) throws Throwable{
String className = point.getTarget().getClass().getName();
String methodName = point.getSignature().getName();
String value = myLog.value();
log.info("类名:{},方法名:{},注解值:{}",className,methodName,value);
log.info("方法之前执行");
long startTime = System.currentTimeMillis();
Object proceed = point.proceed();
long endTime = System.currentTimeMillis();
long time = endTime - startTime;
log.info("方法之后执行");
log.info("方法耗时:{}", time);
return proceed;
}
}
来源:https://blog.csdn.net/qq_27217017/article/details/125582544
猜你喜欢
- SpringBoot集成Mybatis+xml格式的sql配置文件最近一直在研究SpringBoot技术,由于项目需要,必须使用Mybati
- 一、概述1.1 什么是Java工厂模式Java工厂模式是一种创建对象的设计模式,它提供了一种方法,通过该方法可以在不暴露对象创建逻辑的情况下
- 在c#中怎样调用VC写的OCX控件,主要有两个关键环节两个关键环节:导入ocx控件,创建实例。1 注册ocx:regsvr32.exe al
- Android Service 详细介绍:1、Service的概念 2、Service的生命周期 3、实例:控制音乐播放的Service一、
- 官方文档 8.0Spring为不同缓存做了一层抽象,这里通过阅读文档以及源码会对使用以及原理做一些学习笔记。1.简介
- 问题之前项目能够正常运行,因为默认选择db0,后来新的需求来了,不是默认db0,而是给参数选择db。修改后代码如下,却报错NOAUTH Au
- 客户端代码using System;using System.Collections.Generic;using System.Compon
- 区别一如果Mybatis Plus是扳手,那Mybatis Generator就是生产扳手的工厂。通俗来讲——MyBatis:一种操作数据库
- 枚举是 C# 中最有意思的一部分,大部分开发人员只了解其中的一小部分,甚至网上绝大多数的教程也只讲解了枚举的一部分。那么,我将通过这篇文章向
- Java 实现网络爬虫框架最近在做一个搜索相关的项目,需要爬取网络上的一些链接存储到索引库中,虽然有很多开源的强大的爬虫框架,但本着学习的态
- 背景2021年第一天早上,客户突然投诉说系统的一个功能出了问题,紧急排查后发现后端系统确实出了bug,原因为前端传输的JSON报文,后端反序
- 浅谈java内存模型 不同的平台,内存模型是不一样的,但是jvm的
- 如何使用struts2 * ,或者自定义 * 。特别注意,在使用 * 的时候,在Action里面必须最后一定要引用struts2自带的 *
- 比如我创建一个Kotlin Object类:ObjectMethodpackage com.baichuan.example.unit_te
- 本篇主要介绍C#的Excel导入、导出,供大家参考,具体内容如下一. 介绍1.1 第三方类库:NPOI说明:NPOI是POI项目的.NET
- 本文实例为大家分享了UnityShader实现运动模糊的具体代码,供大家参考,具体内容如下原理:像素的当前帧的NDC坐标(x,y
- 对 Excel 进行读写操作是生产环境下常见的业务,网上搜索的实现方式都是基于POI和JXL第三方框架,但都不是很全面。小编由于这两天刚好需
- 近来,很多公司的APP都实现了人脸识别登录的功能。今天呢,银鹏带大家从头到尾做一下这个人脸识别登录。首先呢,我们需要采用一个拥有人脸识别算法
- 格式化一个数值,比如123456789.123,希望显示成"$123,456,789.123".要完成需求,可以用jav
- 冒泡排序冒泡排序的思想: 每次让当前的元素和它的下一个元素比较大小、如果前一个的元素大于后一个元素的话,交换两个元素。这样的话经历一次扫描之