Spring AOP:面向切面编程的最佳实践 ( 一 )

news2025/1/11 8:14:26

1.AOP思想

1.1.为什么需要面向切面编程

如果在一个类或者多个类的多个业务逻辑方法中, 在开始,结尾部分包含功能相同的代码称之为横切关注点也叫切面, 这种结构可能符合传统的面向对象编程(OOP)的封装特性, 但可能导致代码难以维护和扩展。

面向切面编程是一种编程范式。它允许程序员将横切关注点(cross-cutting concerns)从业务逻辑中分离出来, 单独在特殊的类中编写这些功能代码,而原来的业务逻辑中不再编写与之相关的代码, 但依然会对业务逻辑代码产生影响。

在这里插入图片描述

通常这些横切关注点是指那些跨越多个模块或组件的功能,比如日志记录、安全性检查、事务管理等。这样就降低了功能代码与业务逻辑代码的耦合度。

特别指出: AOP 的核心思想是通过预编译或运行时动态代理的方式,在不修改源代码的前提下,对程序动态统一添加额外功能的一种技术。 所以也可能将AOP技术理解为一种方法增加技术。

在 Spring 框架中,AOP 被广泛应用,主要通过 JDK 动态代理和 CGLIB 动态代理实现。Spring AOP 提供了强大的功能来增强 Bean 的行为,使得切面逻辑与核心业务逻辑分离,提升了代码的模块化和可维护性 。

1.2.AOP作用

面向切面编程(AOP是Aspect-Oriented Programming)

我们知道,面向对象的特点是继承、多态和封装。而封装就要求将功能分散到不同的对象中去,这在软件设计中往往称为职责分配。实际上也就是说,让不同的类设计不同的方法。这样代码就分散到一个个的类中去了。这样做的好处是降低了代码的复杂程度,使类可重用。

但是人们也发现,在分散代码的同时,也增加了代码的重复性。什么意思呢?比如说,我们在两个类中,可能都需要在每个方法中做日志。按面向对象的设计方法,我们就必须在两个类的方法中都加入日志的内容。也许他们是完全相同的,但就是因为面向对象的设计让类与类之间无法联系,而不能将这些重复的代码统一起来。

也许有人会说,那好办啊,我们可以将这段代码写在一个独立的类独立的方法里,然后再在这两个类中调用。但是,这样一来,这两个类跟我们上面提到的独立的类就有耦合了,它的改变会影响这两个类。那么,有没有什么办法,能让我们在需要的时候,随意地加入代码呢?这种在运行时,动态地将代码切入到类的指定方法、指定位置上的编程思想就是面向切面的编程。

一般而言,我们管切入到指定类指定方法的代码片段称为切面,而切入到哪些类、哪些方法则叫切入点。有了AOP,我们就可以把几个类共有的代码,抽取到一个切片中,等到需要时再切入对象中去,从而改变其原有的行为。

这样看来,AOP其实只是OOP的补充而已。OOP从横向上区分出一个个的类来,而AOP则从纵向上向对象中加入特定的代码。有了AOP,OOP变得立体了。如果加上时间维度,AOP使OOP由原来的二维变为三维了,由平面变成立体了。从技术上来说,AOP基本上是通过代理机制实现的。

AOP在编程历史上可以说是里程碑式的,对OOP编程是一种十分有益的补充。

2.动态代理与CGLIB代理

在 健身房 类中 多个 方法的开始,结束部分都包含 推荐老师 , 结束提示 等功能代码, 下面我们分别以不同方式实现业务代码与功能代码的分离。

在这里插入图片描述

2.0.准备代码

2.0.1.业务类

这是以 健身房类 为例, 包含 三个方法 分别有参, 有返回值的不同结构。

import java.util.Date;
// 健身房类
public class GymBase {

    // 业务 A : 举重训练
    public String  weightLifting() {
        System.out.println( "首先 : 我们为您推荐一位指导老师!" );
        System.out.println( "进行举重训练。" );
        System.out.println( "您在 " + new Date() + " 结束了" + "weightLifting" );
        return "感觉不断在进步!";
    }

    // 业务 B : 瑜伽课
    public void yogaClass(String type) {
        System.out.println( "首先 : 我们为您推荐一位指导老师!" );
        System.out.println( "参加 " + type + " [ 瑜伽课 ]。");
        System.out.println( "您在 " + new Date() + " 结束了" + "yogaClass" );
    }

    // 业务 C : 游泳训练
    public void swimming() {
        System.out.println( "首先 : 我们为您推荐一位指导老师!" );
        System.out.println( "在游泳池里游泳。");
        System.out.println( "您在 " + new Date() + " 结束了" + "swimming" );
    }
}
2.0.2.测试类
public class TestBase {

    public static void main(String[] args) {

        GymBase gymBase = new GymBase();

        gymBase.yogaClass("综合");
        System.out.println("-----");

        String feel = gymBase.weightLifting();
        System.out.println("您的评价是:" + feel );

        System.out.println("-----");
        gymBase.swimming();
    }
}
2.0.3.运行测试
首先 : 我们为您推荐一位指导老师!
参加 综合 [ 瑜伽课 ]。
您在 Fri Aug 02 21:01:32 CST 2024 结束了yogaClass
-----
首先 : 我们为您推荐一位指导老师!
进行举重训练。
您在 Fri Aug 02 21:01:32 CST 2024 结束了weightLifting
您的评价是:感觉不断在进步!
-----
首先 : 我们为您推荐一位指导老师!
在游泳池里游泳。
您在 Fri Aug 02 21:01:32 CST 2024 结束了swimming

2.1.JDK动态代理

Java提供了一个java.lang.reflect.Proxy类和java.lang.reflect.InvocationHandler接口来实现动态代理:

  1. 定义接口:目标对象需要实现一个或多个接口。
  2. 实现InvocationHandler:创建一个实现了InvocationHandler接口的类,并重写invoke方法,该方法会在代理对象的每个方法调用时执行。
  3. 创建代理对象:使用Proxy.newProxyInstance方法创建代理对象,传入目标对象和InvocationHandler的实例。
  4. 拦截方法调用:在invoke方法中,可以定义拦截逻辑,例如在目标方法执行前后添加日志或其他行为。

通过动态代理实现 AOP 是 Spring AOP 中常用的一种技术。动态代理允许你在运行时创建一个代理对象,这个代理对象可以拦截目标对象的方法调用,并在方法调用前后执行额外的操作。

2.1.1.调整代码增加接口

接口

public interface IGymProxy {

    // 业务 A : 举重训练
    String weightLifting();

    // 业务 B : 瑜伽课
    void yogaClass(String type);

    // 业务 C : 游泳训练
    void swimming();
}

业务实现类 , 每个方法只有核心业务代码

// 健身房类
public class GymProxyImpl implements IGymProxy {

    // 业务 A : 举重训练
    @Override
    public String  weightLifting() {
        System.out.println( "进行举重训练。" );
        return "感觉不断在进步!";
    }

    // 业务 B : 瑜伽课
    @Override
    public void yogaClass(String type) {
        System.out.println( "参加 " + type + " [ 瑜伽课 ]。");
    }

    // 业务 C : 游泳训练
    @Override
    public void swimming() {
        System.out.println( "在游泳池里游泳。");
    }
}

2.1.2.增加切面类
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.util.Arrays;
import java.util.Date;

public class AspectProxy implements InvocationHandler {

    private Object target;

    public Object getTarget() {
        return target;
    }
    public void setTarget(Object target) {
        this.target = target;
    }
    
    //前置的方法
    // 功能 A : 推荐老师
    public void selectMaster(Object[] args) {
        String type = "";
        if (args != null && args.length > 0) {
            type = "根据您的要求: " + Arrays.toString( args );
        }
        System.out.println( "首先 : " + type + " 我们为您推荐一位指导老师!" );
    }

    //后置的方法
    //功能 B : 结束提示
    public void  overTip( Method method ){
        System.out.println( "您在 " + new Date() + " 结束了" + method.getName() );
    }

    @Override
    public Object invoke(Object proxy, Method method, Object[] args)throws Throwable {
        // 功能 A :推荐老师
        selectMaster(args);
        
        // 修改 传入的参数
        if (args != null && args.length > 0) {
            for (int i = 0; i < args.length; i++) {
                args[i] = " { " + args[i] + " } ";
            }
        }
        
        //执行方法
        Object obj=method.invoke(target,args);
        
        // 功能 B : 结束提示
        overTip(method);
        // 修改 返回值
        if (obj!=null) {
            obj = " { " + obj + " } ";
        }
        
        // 方法返回值
        return obj;
    }
}

私有成员变量 target 存储了要被代理的目标对象。通过Setter,Getter两个方法提供了对 target 变量的设置与访问。

重写了invoke方法, 这个方法是 InvocationHandler 接口的核心方法,它会在代理对象的方法被调用时触发。

在方法中 调用另外两个方法来增加功能 , 同时在方法内还 修改了 接收到的参数 及方法执行后的返回值。

2.1.3.测试类
import java.lang.reflect.Proxy;

public class TestProxy {

    public static void main(String[] args) {

        IGymProxy gym = new GymProxyImpl();
        AspectProxy aspectProxy = new AspectProxy();
        aspectProxy.setTarget(gym);

        IGymProxy gymProxy=(IGymProxy) Proxy.newProxyInstance(gym.getClass().getClassLoader(), gym.getClass().getInterfaces(), aspectProxy);

        gymProxy.yogaClass("综合");
        System.out.println("-----");

        String feel = gymProxy.weightLifting();
        System.out.println("您的评价是:" + feel );

        System.out.println("-----");
        gymProxy.swimming();
    }
}

使用 Proxy.newProxyInstance 方法创建代理对象。这个方法需要三个参数:

  • ClassLoader:设置代码使用的类装载器,一般采用跟目标对象相同的类装载器

  • Class<?>[] interfaces:目标对象实现的接口数组。

  • InvocationHandler h:实现了 InvocationHandler 接口的对象,用于处理方法调用。

    ​ 当代理对象的方法被调用时, 会委派给该参数指定对象的invoke方法

2.1.4.运行测试
首先 : 根据您的要求: [综合] 我们为您推荐一位指导老师!
参加  { 综合 }  [ 瑜伽课 ]。
您在 Fri Aug 02 21:27:40 CST 2024 结束了yogaClass
-----
首先 :  我们为您推荐一位指导老师!
进行举重训练。
您在 Fri Aug 02 21:27:40 CST 2024 结束了weightLifting
您的评价是: { 感觉不断在进步! } 
-----
首先 :  我们为您推荐一位指导老师!
在游泳池里游泳。
您在 Fri Aug 02 21:27:40 CST 2024 结束了swimming

2.2.CGLIB代理

CGLIB(Code Generation Library)是一个强大的,高性能,高质量的Code生成类库。它可以在运行期扩展Java类与实现Java接口。

CGLIB包的底层是通过使用一个小而快的字节码处理框架ASM,来转换字节码并生成新的类。

CGLIB用于AOP,jdk中的proxy必须基于接口,CGLIB却没有这个限制。

  1. 使用CGLIB库:CGLIB是一个强大的高性能代码生成库,它可以在运行时扩展Java类和实现接口。
  2. 创建Enhancer对象:使用CGLIB的Enhancer类创建一个增强器对象。
  3. 设置父类:通过setSuperclass方法设置需要被代理的目标类。
  4. 生成代理对象:调用create方法生成代理对象。
2.2.0.导入依赖
        <dependency>
            <groupId>cglib</groupId>
            <artifactId>cglib</artifactId>
            <version>3.3.0</version>
        </dependency>

或者

<!-- https://mvnrepository.com/artifact/cglib/cglib-nodep -->
<dependency>
    <groupId>cglib</groupId>
    <artifactId>cglib-nodep</artifactId>
    <version>3.3.0</version>
</dependency>

cglibcglib-nodep 是 CGLIB 库的不同版本或配置:

  1. cglib:
    • 这是 CGLIB 的标准版本,它可能依赖于其他类库(如 ASM,这是一个字节码操作和分析框架),或者包含了一些额外的功能和组件。
    • 当你使用这个版本时,你需要确保你的项目中没有与其他依赖冲突的问题。
  2. cglib-nodep:
    • “nodep”意味着“no dependencies”,即这个版本没有外部依赖。
    • cglib-nodep 版本包含了所有必要的代码,并且不需要其他类库就能工作。
    • 它是一个独立的版本,可以作为一个单独的 JAR 文件添加到项目中,而不用担心与其他已存在的类库冲突。
    • 这个版本通常用于那些希望避免不必要的依赖关系的项目。
2.2.1.重新修改业务类
// 健身房类
public class GymCglib {

    // 业务 A : 举重训练
    public String  weightLifting() {
        System.out.println( "进行举重训练。" );
        return "感觉不断在进步!";
    }

    // 业务 B : 瑜伽课
    public void yogaClass(String type) {
        System.out.println( "参加 " + type + " [ 瑜伽课 ]。");
    }

    // 业务 C : 游泳训练
    public void swimming() {
        System.out.println( "在游泳池里游泳。");
    }
}

2.2.2.增加切面类
package com.yuan.cglib;

import net.sf.cglib.proxy.Enhancer;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;

import java.lang.reflect.Method;
import java.util.Arrays;
import java.util.Date;

public class AspectCglib implements MethodInterceptor{

    private Object targetObject;

    public Object createProxyIntance(Object targetObject)  {
        this.targetObject = targetObject;
        //该类用于生成代理对象
        Enhancer enhancer = new Enhancer();
        //设置被代理类字节码,CGLIB根据字节码生成被代理类的子类
        enhancer.setSuperclass(this.targetObject.getClass());
        //设置回调用对象为本身
        enhancer.setCallback(this);
        return enhancer.create();
    }

    //前置的方法
    // 功能 A : 推荐老师
    public void selectMaster(Object[] args) {
        String type = "";
        if (args != null && args.length > 0) {
            type = "根据您的要求: " + Arrays.toString( args );
        }
        System.out.println( "首先 : " + type + " 我们为您推荐一位指导老师!" );
    }

    //后置的方法
    //功能 B : 结束提示
    public void  overTip( Method method ){
        System.out.println( "您在 " + new Date() + " 结束了" + method.getName() );
    }

    public Object intercept(Object object, Method method, Object[] args,MethodProxy methodProxy) throws Throwable {

        // 功能 A :  推荐老师
        selectMaster(args);

        // 修改 传入的参数
        if (args != null && args.length > 0) {
            for (int i = 0; i < args.length; i++) {
                args[i] = " { " + args[i] + " } ";
            }
        }

        //执行方法
        Object obj=method.invoke(this.targetObject, args);

        // 功能 B : 结束提示
        overTip(method);

        // 修改 返回值
        if (obj!=null) {
            obj = " { " + obj + " } ";
        }
        
        // 方法返回值
        return obj;
    }
}
2.2.3.测试类
public class TestCglib {

    public static void main(String[] args)  {

        AspectCglib aspectCglib = new AspectCglib();

        GymCglib gymCglib = (GymCglib) aspectCglib.createProxyIntance(new GymCglib());

        gymCglib.yogaClass("综合");
        System.out.println("-----");

        String feel = gymCglib.weightLifting();
        System.out.println("您的评价是:" + feel );

        System.out.println("-----");
        gymCglib.swimming();

    }
}

​ 通过使用AspectCglib的代理创建功能,实例化一个GymCglib的代理对象,
​ 以便在不直接修改GymCglib类代码的情况下,增加额外的功能或行为,实现对GymCglib对象的方法调用的增强。

2.2.4.运行测试
首先 : 根据您的要求: [综合] 我们为您推荐一位指导老师!
参加  { 综合 }  [ 瑜伽课 ]。
您在 Fri Aug 02 21:33:44 CST 2024 结束了yogaClass
-----
首先 :  我们为您推荐一位指导老师!
进行举重训练。
您在 Fri Aug 02 21:33:44 CST 2024 结束了weightLifting
您的评价是: { 感觉不断在进步! } 
-----
首先 :  我们为您推荐一位指导老师!
在游泳池里游泳。
您在 Fri Aug 02 21:33:44 CST 2024 结束了swimming

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1980703.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Python 如何进行自然语言处理(NLTK, SpaCy)

自然语言处理&#xff08;Natural Language Processing, NLP&#xff09;是计算机科学和人工智能的一个重要领域&#xff0c;旨在实现计算机对人类语言的理解和处理。在Python中&#xff0c;有许多工具和库可以用于自然语言处理&#xff0c;其中最流行的两个是NLTK&#xff08;…

【Python脚本】定时任务脚本实现、自动关机等功能脚本实现(保姆篇)

文章目录 功能描述源码分析依赖参数配置数据校验多线程并发执行定时任务注册自动关机主程序 源码整合本篇小结 更多相关内容可查看 功能描述 需要python环境&#xff0c;详情可看主页python相关文章【Python】从0开始写脚本、Selenium详细教程、附源码案例&#xff08;保姆篇&…

3dsMax模型展开UV之后无法删除,3dsmax删除模型上已经展开的UV

3dsmax展开UV之后如何删除UV 方法二 如果不能重置UV通道&#xff0c;在实用工具》更多工具》UVW移除&#xff0c;选中模型&#xff0c;点击UVW移除&#xff0c;移除模型的UVW。

机器人抓取与操作的挑战与进展——挑战赛角度

从竞赛中看机器人抓取与操作的挑战与进展 前言一、国际机器人竞赛有哪些&#xff1f;二、感知方面的挑战与进展二、抓取方面的挑战与进展三、操作方面的挑战与进展总结 前言 本文根据最近的机器人抓取和操作挑战赛 (Robotic Grasping andManipulation Competitions (RGMCs))&a…

OCR图片矫正、表格检测及裁剪综合实践

问题描述 实际工程中&#xff0c;我们经常需要对图片进行预处理&#xff0c;比如&#xff1a; 1、图片是倾斜的 2、图片背景需要处理掉 3、图片的公章需要剔除 4、图片过暗&#xff0c;过亮 5、图片表格检测 6、图片表格版面分析 。。。。。。等等各种情况。 结果展示…

解决PuppeteerSharp生成PDF颜色问题的最佳实践

在现代网络开发中&#xff0c;使用爬虫技术生成PDF文件已成为一种常见需求。然而&#xff0c;开发者经常会遇到一些棘手的问题&#xff0c;其中之一便是使用PuppeteerSharp生成PDF时颜色丢失的问题。本篇文章将概述如何解决这一问题&#xff0c;并提供最佳实践和相关代码示例。…

TI音频功放TAS6511(二)

3.数字音频处理 芯片支持高级数字音频处理能力&#xff0c;包括&#xff1a; 高通滤波器/直流阻断 数字音量控制 PVDD的Foldback/AGL 热Foldback 双象限增益补偿 混合调制 实时负载诊断 低延迟路径 喇叭功率限制 1&#xff09;PVDD Foldback 本功能主要为了防止音频…

学习笔记--算法(双指针)2

复写零 链接&#xff1a;https://leetcode.cn/problems/duplicate-zeros/ 题目 给你一个长度固定的整数数组 arr &#xff0c;请你将该数组中出现的每个零都复写一遍&#xff0c;并将其余的元素向右平移。 注意&#xff1a;请不要在超过该数组长度的位置写入元素。请对输入的…

分享一个基于微信小程序的旅游自助拼团系统(源码、调试、LW、开题、PPT)

&#x1f495;&#x1f495;作者&#xff1a;计算机源码社 &#x1f495;&#x1f495;个人简介&#xff1a;本人 八年开发经验&#xff0c;擅长Java、Python、PHP、.NET、Node.js、Android、微信小程序、爬虫、大数据、机器学习等&#xff0c;大家有这一块的问题可以一起交流&…

想业余时间做个网赚项目,有啥捷径?

前两年我陷入一段迷茫、浮躁、焦虑期。 主要原因是&#xff0c;心很大&#xff0c;力不足。 总想着找到一个高利润、高复购、少竞争的“蓝海”产品。 于是就面临一个尴尬的境地&#xff1a;普通业务看不上&#xff0c;蓝海业务找不着。 而且总想着做推广一步登天&#xff0…

程序员日志之DNF手游女鬼剑前瞻

目录 传送门正文日志1、概要女鬼剑 传送门 SpringMVC的源码解析&#xff08;精品&#xff09; Spring6的源码解析&#xff08;精品&#xff09; SpringBoot3框架&#xff08;精品&#xff09; MyBatis框架&#xff08;精品&#xff09; MyBatis-Plus SpringDataJPA SpringClou…

【教资 · 科目一】综合素质

【科目一】综合素质 单选题&#xff1a;29道/2分&#xff0c;共58分材料分析题&#xff1a;3道/14分&#xff0c;共42分写作题&#xff1a;1道&#xff0c;50分 卷面满分150分 —— 报告满分120分 及格分数99分以上 —— 及格分数70分 ‍ 考试内容&#xff1a; 职业理念&a…

Flink笔记整理(七)

Flink笔记整理&#xff08;七&#xff09; 文章目录 Flink笔记整理&#xff08;七&#xff09;九、容错机制9.1 检查点&#xff08;Checkpoint&#xff09;检查点的保存从检查点恢复状态检查点算法9.2 状态一致性9.3 端到端精确一次&#xff08;End-To-End Exactly-Once&#x…

中科亿海微SoM模组——光纤陀螺控制板

光纤陀螺控制板 光纤陀螺仪是一种高精度、高可靠性的惯性测量仪器&#xff0c;被广泛应用于导航、姿态控制等应用场景&#xff0c;具有非常重要的应用价值。 本文介绍的光纤陀螺控制板是基于中科亿海微自研的SiP芯片平台&#xff0c;以及光纤陀螺数字信号处理流程&#xff0c…

安泰高压功率放大器的作用以及应用有哪些

高压功率放大器是一种用于增强信号强度的电子设备。它的作用是将输入信号的功率放大到足够的水平&#xff0c;以便在系统中进行传输或执行特定的任务。高压功率放大器在各种领域都有广泛的应用&#xff0c;从通信到科学研究再到医疗设备等多个领域都可以找到其身影。 高压功率放…

echarts横向柱状图

一、效果图 二、代码 let option {grid: {top: 8%,bottom: -20,right: 20,left: 20,containLabel: true},xAxis: {show: false},yAxis: [{triggerEvent: true,show: true,inverse: true,data: getArrByKey(data, name),axisLine: {show: false},splitLine: {show: false},axi…

React(五):XLS、XLSX文件在线预览

效果 依赖 $ yarn add xlsx源码 .xlsx-wrap {position: relative;width: 100%;height: 100%;background-color: #fafafa;.ant-tabs {width: 100%;height: 100%;.ant-tabs-nav {height: 50px;padding: 0 10px;margin-bottom: 0;}.ant-tabs-content-holder {border-top: 1px so…

JavaScript对象转数组的三种简单方法

大家好&#xff01;今天我们要聊的是JavaScript中一个非常实用的技巧——将对象转换为数组。 方法1&#xff1a;使用Object.keys()和Array.map() 首先介绍一种基础但非常实用的方法&#xff0c;就是通过Object.keys()获取对象的键&#xff0c;然后用Array.map()把这些键对应的值…

C++ primer plus 第17 章 输入、输出和文件:文件输入和输出01

C primer plus 第17 章 输入、输出和文件&#xff1a;文件输入和输出01 C primer plus 第17 章 输入、输出和文件&#xff1a;文件输入和输出01 文章目录 C primer plus 第17 章 输入、输出和文件&#xff1a;文件输入和输出0117.4 文件输入和输出17.4.1 简单的文件 1/0程序清…

商家转账到零钱分销返佣申请方案及驳回处理办法

分销返佣场景是商家申请最多的场景&#xff0c;因而申请被驳回也是最多的&#xff0c;根据我们上万次成功开通商家转账到零钱的经验&#xff0c;当商家转账到零钱的分销返佣场景被驳回时&#xff0c;按照以下步骤&#xff0c;商家都可以快速过审&#xff1a; 一、分析驳回原因 …