Spring框架深入(二)--AOP面向切面

一、AOP概述

  1、AOP:面向切面编程,实现在不增加代码的基础上,增加一些新的功能(公共功能);

  2、AOP并不是Spring框架持有的,Spring只是支持AOP编程的框架之一,可以整合第三方框架来实现面向切面编程(如:Aspect);

  3、现实的应用场景:使用面向切面编程,AOP框架已经实现了面向切面的很多内容;

  4、程序员使用AOP要做的事情:

    编写公共功能,切面;

    基于AOP框架的配置,直接把核心业务和切面关联起来;

  5、Spring中实现AOP的方式有三种:

    基于AspectJ注解的方式实现

    基于schema的XML配置

    基于ProxyFactoryBean代理实现

 

二、AOP的简单实现_基于AspectJ注解

  编写步骤如下:

  1、创建一个简单的工程,引入jar包:

  

  2、编写核心业务的接口和实现类

public interface ServiceInterface {
    public void sayHello();

    public void sayBye();
}
/*
 * 核心关注点,核心业务
 * */
@Component("service")
public class ServiceImpl implements ServiceInterface {
    @Override
    public void sayHello() {
        // TODO Auto-generated method stub
        System.out.println("say Hello");
    }
    public void sayBye() {
        System.out.println("say Bye");
    }
}

  3、编写一个切面类:我们就简单的做一个打印输出

/*
 * 公共功能:切面
 * */
@Component
@Aspect
public class AdviceMethod {
    //前置通知
    //将这个方法织入核心业务:
    //@Before,表示在核心业务之前织入;
    //execution,表示织入的位置在哪儿
    @Before("point()")
    public void befor() {
        System.out.println("日志");
    }
}

  4、在核心配置文件中声明这个类为切面

<!-- 启用注解 -->
<context:annotation-config></context:annotation-config>
<!-- 扫描 -->
<context:component-scan base-package="service,advice"></context:component-scan>
<!-- 启用AOP aspectj 的配置 -->
<aop:aspectj-autoproxy></aop:aspectj-autoproxy>

  5、最后新建一个测试类

public static void main(String[] args) {
        ApplicationContext ac = new ClassPathXmlApplicationContext("applicationContext.xml");
    //要使用接口来声明核心业务类
    //service就是一个代理对象
    ServiceInterface service = (ServiceInterface) ac.getBean("service");
    //sayHello代码不需要改变,会自动增加日志的功能
    service.sayHello();
    service.sayBye();
}

    注:这儿必须要用接口来声明核心业务类

  这样,一个简单的AOP案例就搭建完成了;

 

三、AOP详解

  利用一种横切的技术,剖析封装的对象的内部,将影响了多个累的行为封装到了一个可重用的模块中,成为Aspect,切面;

  将应用程序中的商业逻辑与其提供支持的通用服务进行分离;

  1、AOP中的常用术语

    (1)切面:公共功能、交叉功能的描述

    (2)通知:实现切面功能的类

    (3)目标对象:被通知的对象,核心关注点的对象

    (4)代理对象:代理的是目标对象,通过代理目标对象就增加了切面功能

      目标对象+切面功能

      实现切面功能最早使用的是代理对象

    (5)连接点:静态概念,代表通知执行的地方

    (6)切入点:动态概念,运行时执行通知的地方,实现切面功能是,连接点就变为切入点;

    (7)引入:静态概念,将切面与目标对象关联起来

    (8)织入:将切面应用到代理对象,是一个过程

  2、通知类型

    前置通知:核心业务执行前执行

    返回后通知:核心业务返回后执行

    异常通知:核心业务发生异常时执行

    后置通知:核心业务执行后执行

    环绕通知:核心任务执行的时候执行

  代码编写如下:使用注解的方式;

@Before("execution(* service.*.*.*())")
public void befor() {
    System.out.println("前置通知,日志");
}
@After("execution(* service.*.*.*())")
public void after() {
    System.out.println("后置通知,执行完成");
}
@AfterReturning("execution(* service.*.*.*())")
public void afterReturn() {
    System.out.println("返回后通知,得到返回值后");
}    
@AfterThrowing("execution(* service.*.*.*())")
public void throwExp() {
    System.out.println("异常通知,异常执行时");
}    
@Around("execution(* service.*.*.*())")
public Object round(ProceedingJoinPoint pjp) throws Throwable {
    System.out.println("环绕前。。。");
    Object o=pjp.proceed();//核心业务的正常执行
    System.out.println("环绕后。。。");
    return o;
}

    执行结果:

    

  3、AspectJ中使用的注解

  (1)、AspectJ是一个面向切面的框架;

  (2)、Spring通过AspectJ实现了一注解的方式定义AOP相关的配置,减少了对于配置文件的依赖

  (3)、AspectJ提供的注解:

    @AspectJ:在类上使用,声明该类为一个切面;

    @Before:在方法上使用,声明该方法的通知类型;

    @After:在方法上使用,声明该方法的通知类型;

    @AfterReturn:在方法上使用,声明该方法的通知类型;

    @AfterThrowing:在方法上使用,声明该方法的通知类型;

    @Around:在方法上使用,声明该方法的通知类型;

    @Pointcut:声明切入点;

@Pointcut("execution(* service.*.*.*())")
public void point() {
}
@Before("point()")
public void befor() {
    System.out.println("前置通知,日志");
}

  4、execution表达式的使用

    AOP配置时,不管是XML配置,注解配置,用于定义pointcut切入点

    execution(* service.*.*.*(..))

    

  5、AOP案例_基于schema

    通过核心配置文件来实现AOP切面类

<!-- AspectJ所提供的基于AOP的配置项 -->
<aop:config>
    <aop:pointcut expression="execution(* service.*.*.*(..))" id="pointCut"/>
    <aop:aspect id="myAspect" ref="adviceMethod_xml">
        <aop:before method="before" pointcut-ref="pointCut"/>
        <aop:after method="after" pointcut-ref="pointCut"/>
        <aop:after-returning method="afterReturn" pointcut-ref="pointCut"/>
        <aop:after-throwing method="throwExp" pointcut-ref="pointCut"/>
        <aop:around method="round" pointcut-ref="pointCut"/>
    </aop:aspect>
</aop:config>

    其实现效果和基于注解一样;

 

四、基于代理实现AOP

  1、Spring的AOP包中提供了实现AOP的各种类

  2、Spring上下文就可以有做AOP的实例,Spring直接基于代理实现AOP

  3、Spring的代理有两种方式

    (1)、JDK动态代理:代理的是接口,目标对象是有接口的

    (2)、CGLIB:代理的就是目标对象本身

  附上从参考文档拿过来的图

  

  

    如果没有代理的时候,就会直接调用pojo.foo();

    如果pojo对象生成了一个与之对应的代理对象,执行的时候,就会调用:代理对象.foo();

  

  4、我们这儿来看一下基于接口代理实现AOP

    (1)、先写一个要代理的接口

/*
 * 要代理的接口
 * */
public interface Subject {
    public void doSomething(String role) ; 
}

    (2)、书写目标对象,实现这个接口

/*
 * 目标对象
 * */
@Component
public class RealSubject implements Subject {
    @Override
    public void doSomething(String role) {
        // TODO Auto-generated method stub
        System.out.println("doSomething...");
    }
}

    (3)、书写一个通知类,这个类需要实现类通知的接口。拦截下目标对象

/*
 * 通知类:实现类通知的接口,该类在Spring的上下文中能够被识别为通知
 * */
@Component
public class Advice implements MethodBeforeAdvice {
    /*
     * 拦截:做权限验证
     * arg0:被调用的方法
     * arg1:给这个方法传递的参数
     * arg2:被代理的对象
     * */
    @Override
    public void before(Method arg0, Object[] arg1, Object arg2) throws Throwable {
        // TODO Auto-generated method stub
        if("admin".equals(arg1[0])) {
            System.out.println("通过验证");
        }else {
            System.out.println("没有通过");
        }
    }
}

    (4)、配置代理对象,也是一个bean,在bean中配置要代理的接口,要实现的通知以及代理的目标对象;

<!-- 
    已有了目标对象realsubject,已有了通知advice
    生成代理对象:代理realSubject,具备advice的功能
    Spring提供了一个类能够帮助我们生成代理对象
 -->
<bean id="proxyFactoryBean" class="org.springframework.aop.framework.ProxyFactoryBean">
    <!-- 要代理的接口 -->
    <property name="proxyInterfaces">
        <list>
            <value>proxyAOP.Subject</value>
        </list>
    </property>
    <!-- 要实现的通知 -->
    <property name="interceptorNames">
        <list>
            <value>advice</value>
        </list>
    </property>
    <!-- 代理的目标对象 -->
    <property name="target" ref="realSubject"></property>
</bean>

    (5)、书写一个测试类。注意,测试类里得到的bean是上面代理的bean;

public static void main(String[] args) {
    ApplicationContext ac = new ClassPathXmlApplicationContext("proxyAOP/applicationContext.xml");
    //基于接口
    Subject subject=(Subject) ac.getBean("proxyFactoryBean");
    subject.doSomething("admin");
}

  这样,一个简单的基于接口代理实现AOP就完成了

 

PS:因作者能力有限,如有误还请谅解

posted @ 2018-05-20 10:50  IVEGOTNOIDEA  阅读(221)  评论(0编辑  收藏  举报