spring揭秘07-aop01-aop基本要素及代理模式3种实现

2024-08-25 02:28

本文主要是介绍spring揭秘07-aop01-aop基本要素及代理模式3种实现,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

  • 【README】
  • 【1】AOP思想演进
    • 【1.1】AOL:面向切面语言
    • 【1.2】AOP实现设想
    • 【1.3】java平台的AOP实现机制
      • 【1.3.1】动态代理(需要目标类实现接口)
      • 【1.3.2】动态字节码增强(不需要目标类实现接口)
      • 【1.3.3】自定义加载器
  • 【2】AOP基本要素
    • 【2.1】Joinpoint切点
    • 【2.2】Pointcut切点表达式
    • 【2.3】Advice通知(横切逻辑)
    • 【2.4】Aspect切面
    • 【2.5】织入器
    • 【2.6】目标对象
  • 【3】代理模式
    • 【3.1】静态代理模式
      • 【3.1.1】静态代理模式的问题
    • 【3.2】JDK动态代理模式
      • 【3.2.1】JDK动态代理模式的问题
    • 【3.3】动态字节码增强(生成)动态代理
    • 【3.4】3种代理模式实现小结

【README】

本文总结自《spring揭秘》,作者王福强,非常棒的一本书,墙裂推荐;

1)业务场景: 需要为每个访问数据库的dao方法新增统计sql执行耗时逻辑; 如何实现? 每个dao方法都调用统计耗时的公共逻辑? 会不会太复杂? 我想,一个正常的技术架构不应该是这样的。 此外,公共逻辑包括但不限于方法执行监控,访问日志收集,数据脱敏等,这些公共逻辑都涉及到如何注入到正常业务逻辑的问题;

  • 解决方法: 把统计sql执行耗时公共逻辑通过代理模式织入到dao方法被调用的上文和下文;
  • 如何实现代理模式? 通过静态代理,还是JDK动态代理,亦或是CGLIB字节码增强动态代理? 对的,这就是引入AOP的目的,即通过代理模式把公共逻辑织入到正常业务功能的上下文,一定程度上可以做到业务代码无侵入织入,实现公共逻辑代码高内聚而不是散落在各个业务方法中

2)AOP: Aspect-Oriented Programming,面向切面编程; AOP中,公共逻辑被抽取为切面, 用户代码通过硬编码或配置或注解把切面逻辑织入到业务功能的上文或下文或者上下文;


【1】AOP思想演进

【1.1】AOL:面向切面语言

与OOP类似,AOP也是一种软件架构概念,它也需要编程语言来实现;

1)AOL面向切面语言:实现AOP的编程语言;AOL语言类型不需要与应用系统编程语言相同;(如应用系统使用java,AOL可以选择AspectC,理论上而言)

2)AOL清单:

  • AspectJ
  • AspectC
  • AspectC++

3)切面织入:把面向切面组件集成(注入)到OOP组件的过程;或者把切面逻辑织入到业务逻辑上下文的过程;

【1.2】AOP实现设想

1)静态AOP: 以最初的AspectJ为代表;底层原理是: 通过特定编译器,把切面逻辑编译成字节码,并把字节码织入到业务class文件形成新的class字节码文件(理解为编译时织入);

  • 优点:没有任何性能损失;
  • 缺点:灵活性不够; 因为有任何变动,都需要修改切面逻辑,重新编译其字节码,重新织入;

2)动态AOP: 在系统运行之后织入切面到切点,而不是在编译时;而且织入信息采用XML文件或其他配置文件保存,运维成本低;若有修改,则修改配置即可;

  • 缺点:有性能损失; 因为动态AOP是在系统运行期间织入切面,会有一定的性能损失(理解为运行时织入 );

【1.3】java平台的AOP实现机制

【1.3.1】动态代理(需要目标类实现接口)

1)动态代理:底层使用JDK动态代理 api,把切面逻辑封装到动态代理的InvocationHandler接口实现类中;

  • 缺点1: 所有需要织入切面逻辑的模块类都需要实现对应接口; 因为动态代理机制仅针对接口有效
  • 缺点2:动态代理在运行时使用反射的运行时织入,相对于编译时织入,有性能损失;
  • 补充:Spring AOP 默认使用JDK动态代理实现AOP

【1.3.2】动态字节码增强(不需要目标类实现接口)

1)动态字节码增强(生成)技术: 使用ASM或CGLIB等java工具库,在程序运行期间为目标对象生成子类,修改子类class文件字节码文件(如新增切面逻辑字节码), 只要修改后的子类class文件满足jvm规范,其就可以正常运行

2)基于动态字节码增强的AOP: 通过动态字节码增强技术针对被织入的类(目标对象类)生成相应子类,并把切面逻辑字节码添加到子类,应用系统在执行期间使用子类bean(代理bean)处理业务逻辑;

3)当目标类没有实现接口, spring 使用动态字节码增强实现AOP

【1.3.3】自定义加载器

1)自定义加载器: 加载器通过读取外部文件规定的织入规则和必要信息,在加载class文件时就把切面逻辑织入到class文件,然后把改动后的class文件交给jvm运行;

  • AspectJ项目的AspectWerkz框架采用的是自定义类加载器实现AOP;
  • 缺点: 某些应用服务器会控制整个类加载器,用户可能无法自定义类加载器的情况;

【2】AOP基本要素

【2.1】Joinpoint切点

1)Joinpoint切点:被织入切面逻辑的程序位置(执行点);

2)切点类型:

  • 方法调用: 当前方法被外部组件调用的程序位置;如构造方法调用;
  • 方法调用执行:当期方法内部执行开始位置到结束位置;
    • 包括构造方法执行,字段设置, 字段获取, 异常处理执行,类初始化(静态代码块初始化);

【2.2】Pointcut切点表达式

1)Pointcut切点表达式: 表示切点位置的表达式;

2)切点表达式类型:

  • 直接指定切点所在方法名称;
  • 正则表达式;
  • 使用特定的Pointcut表述语言;

【2.3】Advice通知(横切逻辑)

1)Advice通知:被织入到切点的横切逻辑(简单理解: 横切逻辑就是切面逻辑,本文认为横切这个术语更加符合aop语义 );

2)横切逻辑类型:

  • Before Advice: 在切点位置之前执行;
  • After Advice: 在切点位置之后执行;
    • After returning Advice: 当切点位置的程序正常执行完成后,才被执行(正常返回);
    • After throwing Advice: 当切点位置的程序抛出异常时,才被执行(抛出异常,非正常返回);
    • After Advice(After Finally Advice): 当切点位置的程序执行正常或异常,都被执行;
  • Around Advice:环绕通知; 在切点位置之前或者之后执行;
  • Introduction:引入通知; 这个非常重要;

3)非引入通知与引入通知区别:

  • 非引入通知:包括Before,After,Around等通知; 以上类型通知是把目标对象(被织入横切逻辑的对象)已有方法作为锚点(载体)织入横切逻辑;(简单理解:纵向织入; 织入横切逻辑到已有方法上下文,织入动作影响已有方法逻辑
  • 引入通知:仅 Introduction,顾名思义,引入的意思是为目标对象织入新方法,它不会把目标对象方法作为锚点 ;(简单理解:横向织入;织入新方法,织入动作不影响已有方法

【2.4】Aspect切面

1)Aspect切面:封装多个Pointcut切点表达式与Advice切面逻辑的实体;

【2.5】织入器

1)spring aop织入器:ProxyFactory类是spring aop最通用的织入器;

2)织入器职责: 把切面逻辑织入到切点;

【2.6】目标对象

1)目标对象: 被织入横切逻辑的对象;



【3】代理模式

1)spring aop底层使用JDK动态代理或动态字节码增强技术把横切逻辑织入到目标对象

2)代理模式有3种实现方式:

  • 静态代理;
  • JDK动态代理;
  • CGLIB动态代理(动态字节码增强(生成)技术)

【3.1】静态代理模式

1)静态代理模式: 需要目标类与代理类都实现相同接口;

【StaticProxyMain】

public class StaticProxyMain {public static void main(String[] args) {new StaticProxyMsgSenderImpl(new YidongMsgSenderImpl()).send("您好,您有待办事项需要处理", "123456");}
}

【StaticProxyMsgSenderImpl】静态代理类 ( 实现接口

public class StaticProxyMsgSenderImpl implements IMsgSender {private IMsgSender msgSendSupport;public StaticProxyMsgSenderImpl(IMsgSender msgSendSupport) {this.msgSendSupport = msgSendSupport;}@Overridepublic void send(String msg, String phoneNum) {System.out.println("static proxy busi: before");msgSendSupport.send(msg, phoneNum);System.out.println("static proxy busi: after");}
}

【IMsgSender】接口

public interface IMsgSender {void send(String msg, String phoneNum);default boolean checkAuth(String phoneNum) {System.out.printf("IMsgSender#checkAuth(): 校验权限; 电话号码:[%s]\n", phoneNum);return false;}
}

【YidongMsgSenderImpl】接口实现类 ( 实现接口

public class YidongMsgSenderImpl implements IMsgSender {@Overridepublic void send(String msg, String phoneNum) {System.out.printf("运营商:[中国移动],短信内容:[%s] ; 电话号码:[%s] \n", msg, phoneNum);}
}

【打印日志】

static proxy busi: before
运营商:[中国移动],短信内容:[您好,您有待办事项需要处理] ; 电话号码:[123456] 
static proxy busi: after

【3.1.1】静态代理模式的问题

1)业务场景: 应用系统不仅有短信发送器(IMsgSender),还有邮件发送器(IEmailSender),两个发送器的发送方法都是send(),且系统监控需求都需要对两个send() 方法做监控; 按照静态代理思想, 岂不是还要为邮件发送器新建一个邮件发送器实现类? 如果还有微信发送器(IWechatSender),那还要再新建一个微信发送器实现类? 累不累?

  • 这就是静态代理模式的局限所在了, 要求每个目标对象的代理对象都要实现对应接口,即便拦截的方法名相同;显然,静态代理模式就不适合做aop ,因为通用性不够

【3.2】JDK动态代理模式

1)基于JDK动态代理API实现动态代理: 对于方法名相同而目标对象不同的代理,仅需要提供1个InvocationHandler接口的实现类;

【JdkDynamicProxyMain】JDK动态代理测试入口main

public class JdkDynamicProxyMain {public static void main(String[] args) {// 发送短信动态代理IMsgSender msgSenderTarget = new YidongMsgSenderImpl();IMsgSender dynamicProxyMsgSender = (IMsgSender) Proxy.newProxyInstance(msgSenderTarget.getClass().getClassLoader(), new Class[]{IMsgSender.class}, new SenderInvocationHandler(msgSenderTarget));String message = "您好,您有待办任务需要处理";String phoneNum = "123456";dynamicProxyMsgSender.checkAuth(phoneNum);dynamicProxyMsgSender.send(message, phoneNum);System.out.println("\n我是分割线=================\n");// 发送邮箱动态代理IEmailSender emailSenderTarget = new GoogleEmailSenderImpl();IEmailSender dynamicProxyEmailSender = (IEmailSender) Proxy.newProxyInstance(emailSenderTarget.getClass().getClassLoader(), new Class[]{IEmailSender.class}, new SenderInvocationHandler(emailSenderTarget));dynamicProxyEmailSender.checkAuth("123@gamil.com");dynamicProxyEmailSender.send(message, "123@gamil.com");}
}

【SenderInvocationHandler】InvocationHandler接口实现类

public class SenderInvocationHandler implements InvocationHandler {private Object target;public SenderInvocationHandler(Object target) {this.target = target;}@Overridepublic Object invoke(Object proxy, Method method, Object[] args) throws Throwable {Object result = null;if ("send".equals(method.getName())) {System.out.println("send() execute : before");result = method.invoke(target, args);System.out.println("send() execute : after");} else {result = method.invoke(target, args);}return result;}
}

【IEmailSender】邮件发送器接口

public interface IEmailSender {void send(String content, String addr);default boolean checkAuth(String addr) {System.out.printf("IEmailSender#checkAuth(): 校验权限; 邮箱地址:[%s]\n", addr);return false;}
}

【GoogleEmailSenderImpl】邮件发送器接口实现类

public class GoogleEmailSenderImpl implements IEmailSender {@Overridepublic void send(String content, String addr) {System.out.printf("发送邮件: 内容=[%s], 邮箱地址=[%s]\n", content, addr);}
}

【打印日志】

IMsgSender#checkAuth(): 校验权限; 电话号码:[123456]
send() execute : before // 横切逻辑上文
运营商:[中国移动],短信内容:[您好,您有待办任务需要处理] ; 电话号码:[123456] 
send() execute : after  // 横切逻辑下文我是分割线=================IEmailSender#checkAuth(): 校验权限; 邮箱地址:[123@gamil.com]
send() execute : before // 横切逻辑上文上文
发送邮件: 内容=[您好,您有待办任务需要处理], 邮箱地址=[123@gamil.com]
send() execute : after  // 横切逻辑下文 

【3.2.1】JDK动态代理模式的问题

1)总结: JDK动态代理解决了静态代理模式中每个目标对象的代理对象都要实现对应接口,即便拦截的方法名相同的问题;

2)JDK动态代理局限性: 所有目标对象(被织入切面逻辑的对象)都需要实现对应接口; 因为动态代理机制仅针对接口有效(即,若目标对象没有实现接口,则无法使用JDK动态代理)

  • 如 YidongMsgSenderImpl 实现IMsgSender接口, GoogleEmailSenderImpl 实现 IEmailSender接口 ;

3)问题:因为JDK动态代理仅针对接口有效,则当应用系统集成的第三方类库存在没有实现接口的类,则这些类就无法通过JDK动态代理实现AOP, 显然,这是不合理的;

  • 解决方法: 使用动态字节码增强技术实现动态代理,即便目标对象或目标类没有实现接口,也可以实现动态代理;

【java.lang.reflect.Proxy】JDK中newProxyInstance定义:

/**
* Params:
loader – the class loader to define the proxy class 
interfaces – the list of interfaces for the proxy class to implement 代理类需要实现的接口列表 
h – the invocation handler to dispatch method invocations to ()  
*/
@CallerSensitivepublic static Object newProxyInstance(ClassLoader loader,Class<?>[] interfaces,InvocationHandler h) {

4)若目标对象不实现接口,则无法使用 JDK动态代理,如下。

【NonInterfaceJdkDynamicProxyMain】目标对象不实现接口导致无法实现JDK动态代理测试main

传入的参数是实现类class数组:new Class[]{GoogleEmailSenderImpl.class} , 而不是接口class数组:new Class[]{IEmailSender.class}

public class NonInterfaceJdkDynamicProxyMain {public static void main(String[] args) {// 发送邮箱动态代理GoogleEmailSenderImpl emailSenderTarget = new GoogleEmailSenderImpl();GoogleEmailSenderImpl dynamicProxyEmailSender = (GoogleEmailSenderImpl) Proxy.newProxyInstance(emailSenderTarget.getClass().getClassLoader(), new Class[]{GoogleEmailSenderImpl.class}, new SenderInvocationHandler(emailSenderTarget));dynamicProxyEmailSender.checkAuth("123@gamil.com");dynamicProxyEmailSender.send("您好,您有待办任务需要处理", "123@gamil.com");}
}

【报错】

Exception in thread "main" java.lang.IllegalArgumentException: com.tom.springnote.chapter08proxypattern.GoogleEmailSenderImpl is not an interface


【3.3】动态字节码增强(生成)动态代理

1)动态字节码增强(生成)技术: 使用ASM或CGLIB等java工具库,在程序运行期间为目标对象生成子类,修改子类class字节码文件(如新增切面逻辑字节码), 只要修改后的子类class文件满足jvm规范,其就可以正常运行

2)基于动态字节码增强的动态代理: 通过动态字节码增强技术针对目标类生成子类,子类通过覆写扩展父类(目标对象类)方法,即把切面逻辑字节码添加到子类(代理类),应用系统运行时使用子类bean(代理类bean)处理业务逻辑;

【CglibDynamicProxyMain】CGLIB动态代理main

public class CglibDynamicProxyMain {public static void main(String[] args) {// 传入targetBusiTarget target = new BusiTarget();Enhancer enhancer = new Enhancer();enhancer.setSuperclass(target.getClass());enhancer.setCallback(new CglibMethodInterceptorImpl(target));BusiTarget dynamicByteProxyTaskSender = (BusiTarget) enhancer.create();dynamicByteProxyTaskSender.send1("您有1个待办任务需要处理", "task001");System.out.println();dynamicByteProxyTaskSender.send2("您有2个待办任务需要处理", "task002");}
}

【BusiTarget】目标类,目标对象所属类,被代理类 ( 目标类没有实现接口 ,也可以实现动态代理

public class BusiTarget {public void send1(String content, String taskId) { System.out.printf("BusiTarget#send1(): 发送任务,内容=[%s], taskId=[%s]\n", content, taskId);}public void send2(String content, String taskId) {System.out.printf("BusiTarget#send2(): 发送任务,内容=[%s], taskId=[%s]\n", content, taskId);}
}

【CglibMethodInterceptorImpl】CGLIB方法拦截器实现类

public class CglibMethodInterceptorImpl implements MethodInterceptor {private final Object target;public CglibMethodInterceptorImpl(Object target) {this.target = target;}@Overridepublic Object intercept(Object object, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {System.out.println("send() execute : before");Object result = method.invoke(target, args);System.out.println("send() execute : after");return result;}
}

【打印日志】

send() execute : before
BusiTarget#send1(): 发送任务,内容=[您有1个待办任务需要处理], taskId=[task001]
send() execute : aftersend() execute : before
BusiTarget#send2(): 发送任务,内容=[您有2个待办任务需要处理], taskId=[task002]
send() execute : after

【3.4】3种代理模式实现小结

1)静态代理模式:通过硬编码方式,传入目标对象构建代理对象; 不灵活,对于每个目标对象(类)都需要新建一个代理类;

2)JDK动态代理模式:针对接口实现代理,目标对象需要实现接口,而代理对象需要实现InvocationHandler接口;与静态代理不同的是,如果多个目标对象的被拦截方法名相同,则仅需要新建一个InvocationHandler实现类; (通过实现接口实现动态代理

3)动态字节码增强动态代理: 动态字节码增强技术针对目标类生成一个子类,子类(代理类)通过覆写扩展父类(目标对象类)方法功能,把子类对象作为代理对象返回给应用系统使用;( 通过继承父类实现动态代理



这篇关于spring揭秘07-aop01-aop基本要素及代理模式3种实现的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1104287

相关文章

Oracle查询优化之高效实现仅查询前10条记录的方法与实践

《Oracle查询优化之高效实现仅查询前10条记录的方法与实践》:本文主要介绍Oracle查询优化之高效实现仅查询前10条记录的相关资料,包括使用ROWNUM、ROW_NUMBER()函数、FET... 目录1. 使用 ROWNUM 查询2. 使用 ROW_NUMBER() 函数3. 使用 FETCH FI

Python脚本实现自动删除C盘临时文件夹

《Python脚本实现自动删除C盘临时文件夹》在日常使用电脑的过程中,临时文件夹往往会积累大量的无用数据,占用宝贵的磁盘空间,下面我们就来看看Python如何通过脚本实现自动删除C盘临时文件夹吧... 目录一、准备工作二、python脚本编写三、脚本解析四、运行脚本五、案例演示六、注意事项七、总结在日常使用

Java实现Excel与HTML互转

《Java实现Excel与HTML互转》Excel是一种电子表格格式,而HTM则是一种用于创建网页的标记语言,虽然两者在用途上存在差异,但有时我们需要将数据从一种格式转换为另一种格式,下面我们就来看看... Excel是一种电子表格格式,广泛用于数据处理和分析,而HTM则是一种用于创建网页的标记语言。虽然两

java图像识别工具类(ImageRecognitionUtils)使用实例详解

《java图像识别工具类(ImageRecognitionUtils)使用实例详解》:本文主要介绍如何在Java中使用OpenCV进行图像识别,包括图像加载、预处理、分类、人脸检测和特征提取等步骤... 目录前言1. 图像识别的背景与作用2. 设计目标3. 项目依赖4. 设计与实现 ImageRecogni

Java中Springboot集成Kafka实现消息发送和接收功能

《Java中Springboot集成Kafka实现消息发送和接收功能》Kafka是一个高吞吐量的分布式发布-订阅消息系统,主要用于处理大规模数据流,它由生产者、消费者、主题、分区和代理等组件构成,Ka... 目录一、Kafka 简介二、Kafka 功能三、POM依赖四、配置文件五、生产者六、消费者一、Kaf

Java访问修饰符public、private、protected及默认访问权限详解

《Java访问修饰符public、private、protected及默认访问权限详解》:本文主要介绍Java访问修饰符public、private、protected及默认访问权限的相关资料,每... 目录前言1. public 访问修饰符特点:示例:适用场景:2. private 访问修饰符特点:示例:

详解Java如何向http/https接口发出请求

《详解Java如何向http/https接口发出请求》这篇文章主要为大家详细介绍了Java如何实现向http/https接口发出请求,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 用Java发送web请求所用到的包都在java.net下,在具体使用时可以用如下代码,你可以把它封装成一

使用Python实现在Word中添加或删除超链接

《使用Python实现在Word中添加或删除超链接》在Word文档中,超链接是一种将文本或图像连接到其他文档、网页或同一文档中不同部分的功能,本文将为大家介绍一下Python如何实现在Word中添加或... 在Word文档中,超链接是一种将文本或图像连接到其他文档、网页或同一文档中不同部分的功能。通过添加超

windos server2022里的DFS配置的实现

《windosserver2022里的DFS配置的实现》DFS是WindowsServer操作系统提供的一种功能,用于在多台服务器上集中管理共享文件夹和文件的分布式存储解决方案,本文就来介绍一下wi... 目录什么是DFS?优势:应用场景:DFS配置步骤什么是DFS?DFS指的是分布式文件系统(Distr

NFS实现多服务器文件的共享的方法步骤

《NFS实现多服务器文件的共享的方法步骤》NFS允许网络中的计算机之间共享资源,客户端可以透明地读写远端NFS服务器上的文件,本文就来介绍一下NFS实现多服务器文件的共享的方法步骤,感兴趣的可以了解一... 目录一、简介二、部署1、准备1、服务端和客户端:安装nfs-utils2、服务端:创建共享目录3、服