什么是AOP

 

AOP(Aspect-OrientedProgramming,面向方面编程),可以说是OOP(Object-Oriented Programing,面向对象编程)的补充和完善。OOP引入封装、继承和多态性等概念来建立一种对象层次结构,用以模拟公共行为的一个集合。当我们需要为分散的对象引入公共行为的时候,OOP则显得无能为力。也就是说,OOP允许你定义从上到下的关系,但并不适合定义从左到右的关系。例如日志功能。日志代码往往水平地散布在所有对象层次中,而与它所散布到的对象的核心功能毫无关系。对于其他类型的代码,如安全性、异常处理和透明的持续性也是如此。这种散布在各处的无关的代码被称为横切(cross-cutting)代码,在OOP设计中,它导致了大量代码的重复,而不利于各个模块的重用。

  

而AOP技术则恰恰相反,它利用一种称为“横切”的技术,剖解开封装的对象内部,并将那些影响了多个类的公共行为封装到一个可重用模块,并将其名为“Aspect”,即方面。所谓“方面”,简单地说,就是将那些与业务无关,却为业务模块所共同调用的逻辑或责任封装起来,便于减少系统的重复代码,降低模块间的耦合度,并有利于未来的可操作性和可维护性。AOP代表的是一个横向的关系,如果说“对象”是一个空心的圆柱体,其中封装的是对象的属性和行为;那么面向方面编程的方法,就仿佛一把利刃,将这些空心圆柱体剖开,以获得其内部的消息。而剖开的切面,也就是所谓的“方面”了。然后它又以巧夺天功的妙手将这些剖开的切面复原,不留痕迹。

  

使用“横切”技术,AOP把软件系统分为两个部分:核心关注点和横切关注点。业务处理的主要流程是核心关注点,与之关系不大的部分是横切关注点。横切关注点的一个特点是,他们经常发生在核心关注点的多处,而各处都基本相似。比如权限认证、日志、事务处理。Aop 的作用在于分离系统中的各种关注点,将核心关注点和横切关注点分离开来。正如Avanade公司的高级方案构架师Adam Magee所说,AOP的核心思想就是“将应用程序中的商业逻辑同对其提供支持的通用服务进行分离。”

  

实现AOP的技术,主要分为两大类:一是采用动态代理技术,利用截取消息的方式,对该消息进行装饰,以取代原有对象行为的执行;二是采用静态织入的方式,引入特定的语法创建“方面”,从而使得编译器可以在编译期间织入有关“方面”的代码。

---------------------

 

代理模式:

一、jar包

spring-framework-2.5.2\dist\spring.jar
spring-framework-2.5.2\lib\log4j\log4j-1.2.14.jar
spring-framework-2.5.2\lib\jakarta-commons\commons-logging.jar
spring-framework-2.5.2\lib\aspectj\aspectjrt.jar
spring-framework-2.5.2\lib\aspectj\aspectjweaver.jar

二、结构

三、代码

ConnHandler.java

package dao;

public class ConnHandler {
    
    //专门处理通用服务
    public void openConn(){
        System.out.println("ConnHandler打开连接");
    }
    
    public void closeConn(){
        System.out.println("ConnHandler关闭连接");
    }
    
}

UserDao.java

package dao;

//user的增删改查
public interface UserDao {

    public void add();
    public void delete();
}

UserDaoImpl.java

package dao;

public class UserDaoImpl implements UserDao {

    @Override
    public void add() {
        System.out.println("------UserDaoimpl中的add方法-----");//商业逻辑
    }

    @Override
    public void delete() {
        System.out.println("------UserDaoimpl中的delete方法-----");
    }
}

Test.java

package test;

import org.springframework.beans.factory.BeanFactory;
import org.springframework.context.support.ClassPathXmlApplicationContext;
import dao.UserDao;

public class Test {
    
    public static void main(String[] args)
    {
        BeanFactory factory = new ClassPathXmlApplicationContext("applicationContext.xml");
        UserDao userDao = (UserDao) factory.getBean("userDao");
        
        userDao.add();
        userDao.delete();
    }
}

applicationContext.xml

<?xml version="1.0" encoding="UTF-8"?>

<!-- 
    AOP需要增加两个JAR包。aspectj里两个jar包
 -->
<beans xmlns="http://www.springframework.org/schema/beans"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xmlns:aop="http://www.springframework.org/schema/aop"
         xmlns:tx="http://www.springframework.org/schema/tx"
         xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.0.xsd
           http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-2.0.xsd
           http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-2.0.xsd">
        
        <!--配置BEAN,定义对象的名字,引用类 -->
        <bean id="userDao" class="dao.UserDaoImpl"/><!-- 义务逻辑 -->
        <bean id="connHandler" class="dao.ConnHandler"/><!-- 通用服务 -->
        
        <!-- 定义了一个方面connAspect其实就是connHandler,*任意返回值,dao包下所有类的所有方法,任意参数 ,情况下应用这个方面-->
        <!-- 执行所有add之前before代码执行前打开连接 (调用openConn这个方法)-->
        <aop:config>
            <aop:aspect id="connAspect" ref="connHandler">
                <aop:pointcut id="addMethod" expression="execution(* dao.*.*(..))"/>
                <aop:before method="openConn" pointcut-ref="addMethod"/>
                <aop:after method="closeConn" pointcut-ref="addMethod"/>
            </aop:aspect>
        </aop:config>
            
</beans>

 

以上既有业务逻辑也有通用服务,而定义aop一个描述,是用配置文件把两者结合,代码更简洁明了。以上既有业务逻辑也有通用服务,而定义aop一个描述,是用配置文件把两者结合,代码更简洁明了。

 --------------------------

 

AOP相关概念

 方面(Aspect):一个关注点的模块化,这个关注点实现可能另外横切多个对象。事务管理是J2EE应用中一个很好的横切关注点例子。方面用Spring的 Advisor或拦截器实现。

  

连接点(Joinpoint): 程序执行过程中明确的点,如方法的调用或特定的异常被抛出。

  

通知(Advice): 在特定的连接点,AOP框架执行的动作。各种类型的通知包括“around”、“before”和“throws”通知。通知类型将在下面讨论。许多AOP框架包括Spring都是以拦截器做通知模型,维护一个“围绕”连接点的拦截器链。Spring中定义了四个advice: BeforeAdvice, AfterAdvice, ThrowAdvice和DynamicIntroductionAdvice

  

切入点(Pointcut): 指定一个通知将被引发的一系列连接点的集合。AOP框架必须允许开发者指定切入点:例如,使用正则表达式。 Spring定义了Pointcut接口,用来组合MethodMatcher和ClassFilter,可以通过名字很清楚的理解, MethodMatcher是用来检查目标类的方法是否可以被应用此通知,而ClassFilter是用来检查Pointcut是否应该应用到目标类上

  

引入(Introduction): 添加方法或字段到被通知的类。 Spring允许引入新的接口到任何被通知的对象。例如,你可以使用一个引入使任何对象实现 IsModified接口,来简化缓存。Spring中要使用Introduction, 可有通过DelegatingIntroductionInterceptor来实现通知,通过DefaultIntroductionAdvisor来配置Advice和代理类要实现的接口

  

目标对象(Target Object): 包含连接点的对象。也被称作被通知或被代理对象。POJO

  

AOP代理(AOP Proxy): AOP框架创建的对象,包含通知。 在Spring中,AOP代理可以是JDK动态代理或者CGLIB代理。

  

织入(Weaving): 组装方面来创建一个被通知对象。这可以在编译时完成(例如使用AspectJ编译器),也可以在运行时完成。Spring和其他纯Java AOP框架一样,在运行时完成织入。