Spring框架第五篇之Spring与AOP

JAVA学习网 2017-07-02 18:18:01

一、AOP概述

AOP(Aspect Orient Programming),面向切面编程,是面向对象编程OOP的一种补充。面向对象编程是从静态角度考虑程序的结构,而面向切面编程是从动态角度考虑程序运行过程。

AOP底层就是采用动态代理模式实现的,采用了两种代理:JDK的动态代理与CGLIB的动态代理。

面向切面编程,就是将交叉业务逻辑封装成切面,利用AOP容器的功能将切面织入到主业务逻辑中。所谓交叉业务逻辑是指,通用的、与主业务逻辑无关的代码。如安全检查、事务、日志等。

若不是用AOP,则会出现代码纠缠,即交叉业务逻辑与主业务逻辑混合在一起,这样会使主业务逻辑变的混杂不清。

二、通知Advice

1、通知详解

(1)前置通知MethodBeforeAdvice

定义前置通知,需要实现MethodBeforeAdvice接口。该接口中有一个方法before(),会在目标方法执行之前执行。

前置通知的特点:

1、在目标方法执行之前执行。

2、不改变目标方法的执行流程,前置通知代码不能阻止目标方法执行。

3、不改变目标方法执行的结果。

 举例:

创建IService接口:

package com.ietree.spring.basic.aop.beforeadvice;

public interface IService {

    void doFirst();

    void doSecond();

}

创建接口的实现类:

package com.ietree.spring.basic.aop.beforeadvice;

public class SomeServiceImpl implements IService {

    @Override
    public void doFirst() {
        System.out.println("执行doFirst()方法");
    }

    @Override
    public void doSecond() {
        System.out.println("执行doFirst()方法");
    }

}

创建前置通知类MyMethodBeforeAdvice,该类必须实现MethodBeforeAdvice接口:

package com.ietree.spring.basic.aop.beforeadvice;

import java.lang.reflect.Method;

import org.springframework.aop.MethodBeforeAdvice;

/**
 * 前置通知
 * 
 * @author Root
 */
public class MyMethodBeforeAdvice implements MethodBeforeAdvice {

    // 当前方法在目标方法执行之前执行
    // method:目标方法
    // args:目标方法参数列表
    // target:目标对象
    @Override
    public void before(Method method, Object[] args, Object target) throws Throwable {
        // 对于目标方法的增强代码就写在这里
        System.out.println("执行前置通知...");
    }

}

配置XML文件:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="
        http://www.springframework.org/schema/beans 
        http://www.springframework.org/schema/beans/spring-beans.xsd">
    
    <!-- 1、注册目标对象 -->
    <bean id="someService" class="com.ietree.spring.basic.aop.beforeadvice.SomeServiceImpl"/>
    
    <!-- 2、注册切面:通知 -->
    <bean id="myAdvice" class="com.ietree.spring.basic.aop.beforeadvice.MyMethodBeforeAdvice"/>
    
    <!-- 3、生成代理对象 -->
    <bean id="serviceProxy" class="org.springframework.aop.framework.ProxyFactoryBean">
        <!-- 指定目标对象 -->
        <!-- <property name="targetName" value="someService"/> -->
        <property name="target" ref="someService"/>
        <!-- 指定切面 -->
        <property name="interceptorNames" value="myAdvice"/>
    </bean>
    
</beans>

测试:

package com.ietree.spring.basic.aop.beforeadvice;

import org.junit.Test;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;

public class MyTest {

    @Test
    public void test() {
        
        String resource = "com/ietree/spring/basic/aop/beforeadvice/applicationContext.xml";
        
        ApplicationContext ac = new ClassPathXmlApplicationContext(resource);
    
        IService service = (IService)ac.getBean("serviceProxy");
        
        service.doFirst();
        System.out.println("==============");
        service.doSecond();
    }

}

输出:

执行前置通知...
执行doFirst()方法
==============
执行前置通知...
执行doFirst()方法

注意:执行之前需要导入spring-aop-4.3.9.RELEASE.jar包

(2)后置通知AfterReturningAdvice

定义前置通知,需要实现AfterReturningAdvice接口。该接口中有一个方法afterReturning(),会在目标方法执行之后执行。

后置通知的特点:

1、在目标方法执行之后执行。

2、不改变目标方法的执行流程,后置通知代码不能阻止目标方法执行。

3、不改变目标方法执行的结果。

大致流程和前置通知差不多,这里就简单列举一下不同之处:

创建后置通知类并实现AfterReturningAdvice接口,重写afterReturning()方法:

package com.ietree.spring.basic.aop.afterreturningadvice;

import java.lang.reflect.Method;

import org.springframework.aop.AfterReturningAdvice;

/**
 * 后置通知:可以获取到目标方法的返回结果,但是无法改变目标方法的结果
 * 
 * @author Root
 */
public class MyAfterReturningAdvice implements AfterReturningAdvice {

    // 在目标方法执行之后执行
    // returnValue:目标方法的返回值
    @Override
    public void afterReturning(Object returnValue, Method method, Object[] args, Object target) throws Throwable {
        System.out.println("执行后置通知方法 returnValue = " + returnValue);
    }

}

配置文件:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="
        http://www.springframework.org/schema/beans 
        http://www.springframework.org/schema/beans/spring-beans.xsd">
    
    <!-- 1、注册目标对象 -->
    <bean id="someService" class="com.ietree.spring.basic.aop.afterreturningadvice.SomeServiceImpl"/>
    
    <!-- 2、注册切面:通知 -->
    <bean id="myAdvice" class="com.ietree.spring.basic.aop.afterreturningadvice.MyAfterReturningAdvice"/>
    
    <!-- 3、生成代理对象 -->
    <bean id="serviceProxy" class="org.springframework.aop.framework.ProxyFactoryBean">
        <!-- 指定目标对象 -->
        <property name="target" ref="someService"/>
        <!-- 指定切面 -->
        <property name="interceptorNames" value="myAdvice"/>
    </bean>
    
</beans>

注意:后置通知可以获取到目标方法的返回结果,但是无法改变目标方法执行的返回结果。

(3)环绕通知MethodInterceptor

 定义环绕通知,需要实现MethodInterceptor接口。环绕通知,也叫方法拦截器,可以在目标方法调用之前及之后做处理,可以改变目标方法的返回值,也可以改变程序执行流程。

创建环绕通知类MyMethodInterceptor,实现MethodInterceptor接口:

package com.ietree.spring.basic.aop.methodinterceptor;

import org.aopalliance.intercept.MethodInterceptor;
import org.aopalliance.intercept.MethodInvocation;

// 环绕通知:可以修改目标方法的返回结果
public class MyMethodInterceptor implements MethodInterceptor {

    @Override
    public Object invoke(MethodInvocation invocation) throws Throwable {
        
        System.out.println("执行环绕通知:目标方法执行之前");
        // 执行目标方法
        Object result = invocation.proceed();
        
        System.out.println("执行环绕通知:目标方法执行之后");
        if(null != result)
        {
            result = ((String)result).toUpperCase();
        }
        return result;
    }

}

配置文件:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="
        http://www.springframework.org/schema/beans 
        http://www.springframework.org/schema/beans/spring-beans.xsd">
    
    <!-- 1、注册目标对象 -->
    <bean id="someService" class="com.ietree.spring.basic.aop.methodinterceptor.SomeServiceImpl"/>
    
    <!-- 2、注册切面:通知 -->
    <bean id="myAdvice" class="com.ietree.spring.basic.aop.methodinterceptor.MyMethodInterceptor"/>
    
    <!-- 3、生成代理对象 -->
    <bean id="serviceProxy" class="org.springframework.aop.framework.ProxyFactoryBean">
        <!-- 指定目标对象 -->
        <!-- <property name="targetName" value="someService"/> -->
        <property name="target" ref="someService"/>
        <!-- 指定切面 -->
        <property name="interceptorNames" value="myAdvice"/>
    </bean>
    
</beans>

测试:

package com.ietree.spring.basic.aop.methodinterceptor;

import org.junit.Test;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;

public class MyTest {

    @Test
    public void test() {
        
        String resource = "com/ietree/spring/basic/aop/methodinterceptor/applicationContext.xml";
        
        ApplicationContext ac = new ClassPathXmlApplicationContext(resource);
    
        IService service = (IService)ac.getBean("serviceProxy");
        
        service.doFirst();
        System.out.println("==============");
        
        String result = service.doSecond();
        System.out.println(result);
    }

}

输出:

执行环绕通知:目标方法执行之前
执行doFirst()方法
执行环绕通知:目标方法执行之后
==============
执行环绕通知:目标方法执行之前
执行doSecond()方法
执行环绕通知:目标方法执行之后
ABCDE

注意:环绕通知不仅可以获取到方法的返回结果,而且还可以修改方法的返回结果。

(4)异常通知ThrowsAdvice

异常分两种:
1)运行时异常,不进行处理,也可以通过编译。若一个类继承自RunTimeException,则该异常就是运行时异常
 2)编译时异常,受查异常,Checked Exception。不进行处理,则无法通过编译。若一个类继承自Exception,则该异常就是受查异常。

创建接口IService:

package com.ietree.spring.basic.aop.throwsadvice;

public interface IService {
    // 用户登录
    boolean login(String username, String password) throws UserException;
}

创建接口实现类SomeServiceImpl:

package com.ietree.spring.basic.aop.throwsadvice;

public class SomeServiceImpl implements IService {

    @Override
    public boolean login(String username, String password) throws UserException {
        if (!"jack".equals(username)) {
            throw new UsernameException("用户名错误!");
        }
        if (!"123".equals(password)) {
            throw new PasswordException("密码错误!");
        }
//        double i = 3 / 0;
        return true;
    }

}

创建三个自定义异常:

package com.ietree.spring.basic.aop.throwsadvice;

/**
 * 自定义异常
 * 异常分两种:
 * 1)运行时异常,不进行处理,也可以通过编译。若一个类继承自RunTimeException,则该异常就是运行时异常
 * 2)编译时异常,受查异常,Checked Exception。不进行处理,则无法通过编译。若一个类继承自Exception,则该异常就是受查异常。
 * 
 * @author Root
 */
public class UserException extends Exception {

    public UserException() {
        super();
    }

    public UserException(String message) {
        super(message);
    }

}

用户名异常:

package com.ietree.spring.basic.aop.throwsadvice;

public class UsernameException extends UserException {

    public UsernameException() {
        super();
    }

    public UsernameException(String message) {
        super(message);
    }

}

密码异常:

package com.ietree.spring.basic.aop.throwsadvice;

public class PasswordException extends UserException {

    public PasswordException() {
        super();
    }

    public PasswordException(String message) {
        super(message);
    }

}

定义异常通知:

package com.ietree.spring.basic.aop.throwsadvice;

import org.springframework.aop.ThrowsAdvice;

/**
 * 异常通知 当目标方法抛出与指定类型的异常具有is-a关系的异常时,执行当前方法afterThrowing()
 * 
 * @author Root
 */
public class MyThrowsAdvice implements ThrowsAdvice {

    // 当目标方法抛出UsernameException异常时,执行当前方法
    public void afterThrowing(UsernameException ex) {
        System.out.println("发生用户名异常 ex = " + ex.getMessage());
    }

    // 当目标方法抛出UsernameException异常时,执行当前方法
    public void afterThrowing(PasswordException ex) {
        System.out.println("发生密码异常 ex = " + ex.getMessage());
    }

    // 当目标方法抛出UsernameException异常时,执行当前方法
    public void afterThrowing(Exception ex) {
        System.out.println("发生其它异常 ex = " + ex.getMessage());
    }
}

配置文件:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="
        http://www.springframework.org/schema/beans 
        http://www.springframework.org/schema/beans/spring-beans.xsd">
    
    <!-- 1、注册目标对象 -->
    <bean id="someService" class="com.ietree.spring.basic.aop.throwsadvice.SomeServiceImpl"/>
    
    <!-- 2、注册切面:通知 -->
    <bean id="myAdvice" class="com.ietree.spring.basic.aop.throwsadvice.MyThrowsAdvice"/>
    
    <!-- 3、生成代理对象 -->
    <bean id="serviceProxy" class="org.springframework.aop.framework.ProxyFactoryBean">
        <!-- 指定目标对象 -->
        <!-- <property name="targetName" value="someService"/> -->
        <property name="target" ref="someService"/>
        <!-- 指定切面 -->
        <property name="interceptorNames" value="myAdvice"/>
    </bean>
    
</beans>

测试:

package com.ietree.spring.basic.aop.throwsadvice;

import org.junit.Test;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;

public class MyTest {

    @Test
    public void test() throws UserException {
        
        String resource = "com/ietree/spring/basic/aop/throwsadvice/applicationContext.xml";
        
        ApplicationContext ac = new ClassPathXmlApplicationContext(resource);
    
        IService service = (IService)ac.getBean("serviceProxy");
        
        service.login("jack", "123");
    }

}

 (5)同时使用多个通知的配置方法

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="
        http://www.springframework.org/schema/beans 
        http://www.springframework.org/schema/beans/spring-beans.xsd">
    
    <!-- 1、注册目标对象 -->
    <bean id="someService" class="com.ietree.spring.basic.aop.multipleadvice.SomeServiceImpl"/>
    
    <!-- 2、注册切面:通知 -->
    <!-- 前置通知 -->
    <bean id="beforeAdvice" class="com.ietree.spring.basic.aop.multipleadvice.MyMethodBeforeAdvice"/>
    <!-- 后置通知 -->
    <bean id="afterAdvice" class="com.ietree.spring.basic.aop.multipleadvice.MyAfterReturningAdvice"/>
    
    <!-- 3、生成代理对象 -->
    <bean id="serviceProxy" class="org.springframework.aop.framework.ProxyFactoryBean">
        <!-- 指定目标对象 -->
        <property name="target" ref="someService"/>
        <!-- 指定切面 -->
        <!-- 方式一 -->
        <property name="interceptorNames" value="beforeAdvice,afterAdvice"/>
        <!-- 方式二 -->
        <!-- <property name="interceptorNames">
            <array>
                <value>beforeAdvice</value>
                <value>afterAdvice</value>
            </array>
        </property> -->
    </bean>
    
</beans>

 

三、顾问Advisor

 

四、自动代理生成器

 

五、AspectJ对AOP的实现

 

阅读(811) 评论(0)