`
凤凰山
  • 浏览: 148368 次
  • 性别: Icon_minigender_1
  • 来自: 重庆
社区版块
存档分类
最新评论

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 [add]begin with args ([10.0, 10.0])
the method [add]end 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.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();

        /*调用add方法*/
        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)

 


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

  

 信仰や欺骗 发表于: ttp://www.cnblogs.com/beliefbetrayal/archive/2012/02/03/2337522.html

分享到:
评论

相关推荐

    AOP面向切面编程总结

    ### AOP面向切面编程详解 #### 一、AOP概览 AOP(Aspect-Oriented Programming,面向切面编程)是一种编程思想和技术,它作为OOP(面向对象编程)的一种补充,主要解决了OOP在处理横切关注点方面的不足。在传统的...

    AOP面向切面编程.ppt

    AOP面向切面编程.ppt

    aop 面向切面编程 demo

    面向切面编程(AOP)是一种编程范式,它旨在将关注点分离,使得系统中的核心业务逻辑与系统服务(如日志、事务管理、权限控制等)可以解耦。在Android开发中,AOP的应用可以帮助我们更好地组织代码,提高可维护性和...

    JAVA Spring AOP面向切面编程笔记

    JAVA Spring AOP面向切面编程笔记

    AOP面向切面编程实例

    面向切面编程(Aspect-Oriented Programming,简称AOP)是一种编程范式,它旨在提高软件的模块化程度,通过将关注点分离到不同的“切面”中来实现。在.NET环境中,AOP通常通过拦截器(Interceptor)或动态代理...

    Spring-aop面向切面编程实例

    面向切面编程(Aspect-Oriented Programming,AOP)是Spring框架的核心特性之一,它提供了一种优雅的方式来处理系统的横切关注点,如日志、事务管理、性能监控和权限控制等。在Spring中,AOP主要通过代理模式实现,...

    .net AOP 面向切面编程很好的例子

    spect Oriented Programming(AOP),面向切面编程,是一个比较热门的话题。AOP主要实现的目的是针对业务处理过程中的切面进行提取,它所面对的是处理过程中的某个步骤或阶段,以获得逻辑过程中各部分之间低耦合性的...

    SpringBoot之AOP面向切面编程实例.rar

    在IT行业中,Spring Boot是一个广泛使用的Java框架,它极大地简化了Spring应用...这就是Spring Boot中AOP面向切面编程的基本原理和实践方法。通过不断地学习和实践,开发者可以更好地利用AOP来提高代码质量和可维护性。

    Spring4AOP 面向切面编程实例之方法拦截

    Spring4AOP 面向切面编程实例之方法拦截实例 一下利用Spring4的最后一个版本Spring4.3.9,实现简单的方法拦截实例。 Eclipse 建立java工程,导入必要的jar包,工程目录如下:

    Android之AOP面向切面编程Demo

    Android之AOP面向切面编程Demo 详情文档 http://note.youdao.com/noteshare?id=d522702b72433b59ae630b3922e664e2&sub=A5567A5C7E3E4486B3E9158153C4DFA7

    Spring框架资料,Ioc容器, AOP面向切面编程 ,声明式事务 ,Spring5新特性

    Spring框架资料,Ioc容器, AOP面向切面编程 ,声明式事务 ,Spring5新特性

    Java Spring AOP 面向切面编程的4种实现

    动态代理是一种在运行时创建代理对象的技术,通常用于实现AOP(面向切面编程)中的横切关注点,比如日志、性能监控等。以下是几种常见的动态代理实现: JDK动态代理: Java通过java.lang.reflect.Proxy类提供了一种...

    AOP面向切面编程.docx

    面向切面编程(AOP)是一种编程范式,旨在提高软件设计的模块化和可维护性。它通过将关注点分离,使得系统中的横切关注点(如日志、事务管理、安全检查等)可以从核心业务逻辑中解耦。AOP在Java中主要通过动态代理来...

    AOP面向切面编程的JavaScript实现

    面向切面编程(Aspect-Oriented Programming,AOP)是一种编程范式,旨在将关注点分离,使得系统中的交叉关注点,如日志、事务管理、权限检查等,可以从核心业务逻辑中解耦出来。在Java领域,Spring框架是AOP的典型...

    CH03-AOP面向切面编程.pptx

    AOP(Aspect Oriented Programming):面向切面编程,通过预编译方式和运行期动态代理实现程序功能的统一维护的一种技术。 AOP是对OOP的补充,是软件开发中的一个热点,也是Spring框架中的一个重要内容。 利用AOP...

    Java实现AOP面向切面编程的实例教程共4页.pdf.z

    面向切面编程(Aspect-Oriented Programming,AOP)是软件开发中的一种设计模式,它旨在提高代码的可重用性、模块化和可维护性。在Java中,AOP通常通过Spring框架来实现,Spring提供了强大的AOP支持,允许开发者定义...

    AOP 面向切面编程1

    面向切面编程(AOP)是一种编程范式,旨在提高软件设计的模块化程度,将关注点分离到单独的部分,称为“切面”。这样可以更好地管理横切关注点,如日志、事务管理、安全检查等,这些关注点通常会分散在应用程序的...

    spring4 AOP 面向切面编程@Aspect

    在Spring框架中,AOP(面向切面编程)是一种强大的设计模式,它允许开发者将关注点从业务逻辑中分离出来,比如日志记录、事务管理、权限检查等。`@Aspect`是Spring AOP的核心注解,用于定义一个切面。下面我们将详细...

Global site tag (gtag.js) - Google Analytics