AOP面向切面编程.docx_第1页
AOP面向切面编程.docx_第2页
AOP面向切面编程.docx_第3页
AOP面向切面编程.docx_第4页
AOP面向切面编程.docx_第5页
已阅读5页,还剩6页未读 继续免费阅读

下载本文档

版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领

文档简介

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; /*示意代码 暂时不考虑除数0的情况*/ 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; /*示意代码 暂时不考虑除数0的情况*/ 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); /*同上*/ /*示意代码 暂时不考虑除数0的情况*/ 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可以使用代理模式来实现。代理模式的原理是使用一个代理将对象包装起来,然后用该代理对象取代原始的对象,任何对原始对象的调用首先要经过代理。代理对象负责决定是否以及何时将方法调用信息转发到原始对象上。与此同时,围绕着每个方法的调用,代理对象也可以执行一些额外的工作。可以看出代理模式非常适合实现横切关注点。由于本人只了解Java,所以姑且认为代理模式有两种实现方式,一种是静态代理、另一种是动态代理。他们的区别在于编译时知不知道代理的对象是谁。在模块比较多的系统中,静态代理是不合适也非常低效的,因为静态代理需要专门为每一个接口设计一个代理类,系统比较大成百上千的接口是很正常的,静态代理模式太消耗人力了。动态代理是JDK所支持的代理模式,它可以非常好的实现横切关注点。/*使用动态代理需要实现InvocationHandler接口*/public class ArithmeticCalculatorInvocationHandler implements InvocationHandler /*要代理的对象,动态代理只有在运行时才知道代理谁,所以定义为Object类型,可以代理任意对象*/ private Object target = null; /*通过构造函数传入原对象*/ public ArithmeticCalculatorInvocationHandler(Object target) this.target = target; /*InvocationHandler接口的方法,proxy表示代理,method表示原对象被调用的方法,args表示方法的参数*/ 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(); /*调用add方法*/ arithmeticCalculatorProxy.add(10, 10); 复制代码控制台的输出:the method addbegin with args (10.0, 10.0)the method addend with result (20.0)复制代码可以看到使用动态代理实现了横切关注点。 若需要添加参数验证功能,只需要再创建一个参数验证代理即可:public class ArithmeticCalculatorArgsInvocationHandler implements InvocationHandler /*要代理的对象,动态代理只有在运行时才知道代理谁,所以定义为Object类型,可以代理任意对象*/ private Object target = null; /*通过构造函数传入原对象*/ public ArithmeticCalculatorArgsInvocationHandler(Object target) this.target = target; /*InvocationHandler接口的方法,proxy表示代理,method表示原对象被调用的方法,args表示方法的参数*/ 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(); /*调用add方法*/ argValidatorProxy.add(10, 10); 复制代码控制台输出:begin valid method add with args 10.0, 10.0the method addbegin with args (10.0, 10.0)the method addend with result (20.0)复制代码输入一个负数数据:public class Client public static void main(String args) throws Exception /*获得代理*/ Calculator arithmeticCalculatorProxy = (Calculator)new ArithmeticCalculatorInvocationHandler( new ArithmeticCalculator()

温馨提示

  • 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
  • 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
  • 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
  • 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
  • 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
  • 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
  • 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

最新文档

评论

0/150

提交评论