文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

如何进行AOP的应用分析

2023-06-17 11:06

关注

这篇文章将为大家详细讲解有关如何进行AOP的应用分析,文章内容质量较高,因此小编分享给大家做个参考,希望大家阅读完这篇文章后对相关知识有一定的了解。

AOP(Aspect-Oriented Programming,面向切面的编程),它是可以通过预编译方式和运行期动态代理实现在不修改源代码的情况下给程序动态统一添加功能的一种技术。它是一种新的方法论,它是对传统OOP编程的一种补充。

OOP是关注将需求功能划分为不同的并且相对独立,封装良好的类,并让它们有着属于自己的行为,依靠继承和多态等来定义彼此的关系;AOP是希望能够将通用需求功能从不相关的类当中分离出来,能够使得很多类共享一个行为,一旦发生变化,不必修改很多类,而只需要修改这个行为即可。

AOP是使用切面(aspect)将横切关注点模块化,OOP是使用类将状态和行为模块化。在OOP的世界中,程序都是通过类和接口组织的,使用它们实现程序的核心业务逻辑是十分合适。但是对于实现横切关注点(跨越应用程序多个模块的功能需求)则十分吃力,比如日志记录,验证。

 public interface Calculator  {      public double add(double num1, double num2) throws Exception;      public double sub(double num1, double num2) throws Exception;      public double div(double num1, double num2) throws Exception;      public double mul(double num1, double num2) throws Exception;  }
 public class ArithmeticCalculator implements Calculator  {      @Override     public double add(double num1, double num2)      {          double result = num1 + num2;          return result;      }       @Override     public double sub(double num1, double num2)      {          double result = num1 - num2;          return result;      }            @Override     public double div(double num1, double num2)      {          double result = num1 / num2;          return result;      }       @Override     public double mul(double num1, double num2)      {          double result = num1 * num2;          return result;      }  }

大多数应用程序都有一个通用的需求,即在程序运行期间追踪正在发生的活动。为了给计算机添加日志功能,ArithmeticCalculator类改变如下:

 public class ArithmeticCalculator implements Calculator  {      @Override     public double add(double num1, double num2)      {          System.out.println("the method [add()]"+"begin with args ("+num1+","+num2+")");          double result = num1 + num2;          System.out.println("the method [add()]"+"end with result ("+result+")");                    return result;      }       @Override     public double sub(double num1, double num2)      {          System.out.println("the method [sub()]"+"begin with args ("+num1+","+num2+")");          double result = num1 - num2;          System.out.println("the method [sub()]"+"end with result ("+result+")");                    return result;      }            @Override     public double div(double num1, double num2)      {          System.out.println("the method [div()]"+"begin with args ("+num1+","+num2+")");          double result = num1 / num2;          System.out.println("the method [div()]"+"end with result ("+result+")");                    return result;      }       @Override     public double mul(double num1, double num2)      {          System.out.println("the method [mul()]"+"begin with args ("+num1+","+num2+")");          double result = num1 * num2;          System.out.println("the method [mul()]"+"end with result ("+result+")");                    return result;      }  }

若ArithmeticCalculator规定只能计算正数时,又需要添加参数验证方法:

 public class ArithmeticCalculator implements Calculator  {      @Override     public double add(double num1, double num2) throws Exception      {          this.argsValidatior(num1);          this.argsValidatior(num2);                          }       @Override     public double sub(double num1, double num2) throws Exception      {          this.argsValidatior(num1);          this.argsValidatior(num2);                          }            @Override     public double div(double num1, double num2) throws Exception      {          this.argsValidatior(num1);          this.argsValidatior(num2);                          }       @Override     public double mul(double num1, double num2) throws Exception      {          this.argsValidatior(num1);          this.argsValidatior(num2);                         }            private void argsValidatior(double arg)throws Exception      {          if(arg < 0)              throw new Exception("参数不能为负数");      }  }

上面的程序一个很直观的特点就是,好多重复的代码,并且当加入越来越多的非业务需求(例如日志记录和参数验证),原有的计算器方法变得膨胀冗长。这里有一件非常痛苦的事情,无法使用原有的编程方式将他们模块化,从核心业务中提取出来。例如日志记录和参数验证,AOP里将他们称为横切关注点(crosscutting concern),它们属于系统范围的需求通常需要跨越多个模块。

在使用传统的面向对象的编程方式无法理想化的模块化横切关注点,程序员不能不做的就是将这些横切关注点放置在每一个模块里与核心逻辑交织在一起,这将会导致横切关注点在每一个模块里到处存在。使用非模块化的手段实现横切关注将会导致,代码混乱,代码分散,代码重复。你想想看如果日志记录需要换一种显示方式,那你要改多少代码,一旦漏掉一处(概率很高),将会导致日志记录不一致。这样的代码很维护。种种原因表明,模块只需要关注自己原本的功能需求,需要一种方式来将横切关注点冲模块中提取出来。

忍无可忍的大牛们提出了AOP,它是一个概念,一个规范,本身并没有设定具体语言的实现,也正是这个特性让它变的非常流行,现在已经有许多开源的AOP实现框架了。本次不是介绍这些框架的,我们将不使用这些框架,而是使用底层编码的方式实现最基本的AOP解决上面例子出现的问题。AOP实际是GoF设计模式的延续,设计模式孜孜不倦追求的是调用者和被调用者之间的解耦,AOP可以说也是这种目标的一种实现。AOP可以使用"代理模式"来实现。

如何进行AOP的应用分析

代理模式的原理是使用一个代理将对象包装起来,然后用该代理对象取代原始的对象,任何对原始对象的调用首先要经过代理。代理对象负责决定是否以及何时将方法调用信息转发到原始对象上。与此同时,围绕着每个方法的调用,代理对象也可以执行一些额外的工作。可以看出代理模式非常适合实现横切关注点。

由于本人只了解Java,所以姑且认为代理模式有两种实现方式,一种是静态代理、另一种是动态代理。他们的区别在于编译时知不知道代理的对象是谁。在模块比较多的系统中,静态代理是不合适也非常低效的,因为静态代理需要专门为每一个接口设计一个代理类,系统比较大成百上千的接口是很正常的,静态代理模式太消耗人力了。动态代理是JDK所支持的代理模式,它可以非常好的实现横切关注点。

 public class ArithmeticCalculatorInvocationHandler implements InvocationHandler  {           private Object target = null;                 public ArithmeticCalculatorInvocationHandler(Object target)      {          this.target = target;      }            @Override     public Object invoke(Object proxy, Method method, Object[] args)              throws Throwable      {                   System.out.println("the method ["+method.getName()+"]"+"begin with args ("+Arrays.toString(args)+")");                    Object result = method.invoke(this.target, args);                             System.out.println("the method ["+method.getName()+"]"+"end with result ("+result+")");                    return result;      }                 public Object getProxy()      {          return Proxy.newProxyInstance(this.target.getClass().getClassLoader(), this.getClass().getInterfaces(), this);      }  }

场景类调用:

public class Client  {      public static void main(String[] args) throws Exception      {                   Calculator arithmeticCalculatorProxy = (Calculator)new ArithmeticCalculatorInvocationHandler(                   new ArithmeticCalculator()).getProxy();                    arithmeticCalculatorProxy.add(10, 10);      }  }

控制台的输出:

the method [add]begin with args ([10.0, 10.0])  the method [add]end with result (20.0)

可以看到使用动态代理实现了横切关注点。

如何进行AOP的应用分析

若需要添加参数验证功能,只需要再创建一个参数验证代理即可:

public class ArithmeticCalculatorArgsInvocationHandler implements         InvocationHandler  {           private Object target = null;                 public ArithmeticCalculatorArgsInvocationHandler(Object target)      {          this.target = target;      }            @Override     public Object invoke(Object proxy, Method method, Object[] args)              throws Throwable      {          System.out.println("begin valid method ["+method.getName()+"] with args "+Arrays.toString(args));                    for(Object arg : args)          {              this.argValidtor((Double)arg);          }                    Object result = method.invoke(this.target, args);                    return result;      }                 public Object getProxy()      {          return Proxy.newProxyInstance(this.target.getClass().getClassLoader(), this.target.getClass().getInterfaces(), this);      }            private void argValidtor(double arg) throws Exception      {          if(arg < 0)              throw new Exception("参数不能为负数!");      }  }

场景类调用:

public class Client  {      public static void main(String[] args) throws Exception      {                   Calculator arithmeticCalculatorProxy = (Calculator)new ArithmeticCalculatorInvocationHandler(                   new ArithmeticCalculator()).getProxy();                    Calculator argValidatorProxy = (Calculator)new ArithmeticCalculatorArgsInvocationHandler(arithmeticCalculatorProxy).getProxy();                    argValidatorProxy.add(10, 10);      }  }

控制台输出:

begin valid method [add] with args [10.0, 10.0]  the method [add]begin with args ([10.0, 10.0])  the method [add]end with result (20.0)

输入一个负数数据:

public class Client  {      public static void main(String[] args) throws Exception      {                   Calculator arithmeticCalculatorProxy = (Calculator)new ArithmeticCalculatorInvocationHandler(                   new ArithmeticCalculator()).getProxy();                    Calculator argValidatorProxy = (Calculator)new ArithmeticCalculatorArgsInvocationHandler(arithmeticCalculatorProxy).getProxy();                    argValidatorProxy.add(-10, 10);      }  }

控制台输出:

begin valid method [add] with args [-10.0, 10.0]  Exception in thread "main" java.lang.Exception: 参数不能为负数!      at com.beliefbetrayal.aop.ArithmeticCalculatorArgsInvocationHandler.argValidtor(ArithmeticCalculatorArgsInvocationHandler.java:46)      at com.beliefbetrayal.aop.ArithmeticCalculatorArgsInvocationHandler.invoke(ArithmeticCalculatorArgsInvocationHandler.java:29)      at $Proxy0.add(Unknown Source)      at com.beliefbetrayal.aop.Client.main(Client.java:14)

如何进行AOP的应用分析


不知道你有没有使用过Struts2,这个结构和Struts2的拦截器非常相似,一个个Action对象好比我们的原对象业务核心,一个个拦截器好比是这里的代理,通用的功能实现成拦截器,让Action可以共用,Struts2的拦截器也是AOP的优秀实现。

关于如何进行AOP的应用分析就分享到这里了,希望以上内容可以对大家有一定的帮助,可以学到更多知识。如果觉得文章不错,可以把它分享出去让更多的人看到。

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯