天天看点

Spring-AOP 增强(Advice)5种类型和创建增强类概述增强类型提示

文章目录

  • 概述
  • 增强类型
    • 前置增强
      • 实例一:通过代码实现增强
      • 实例二 通过配置文件实现前置增强-ProxyFactory
    • 后置增强
    • 环绕增强
    • 异常抛出增强
    • 引介增强
  • 提示

概述

Spring使用增强类定义横切逻辑,同时由于Spring只支持方法连接点,增强还包括在方法的哪一点加入横切代码的方位信息,所以增强既包括横切逻辑,又包括部分连接点的信息

增强类型

AOP联盟为增强定义了org.aopalliance.aop.Advice接口。

先了解一下增强接口的继承关系

Spring-AOP 增强(Advice)5种类型和创建增强类概述增强类型提示
  • 带Spring标志的是Spring定义的扩展增强接口
  • 带aopalliance标志的是AOP联盟所定义的接口

按照增强在目标类方法连接点的位置可以将增强划分为以下五类:

  1. 前置增强 (org.springframework.aop.BeforeAdvice) 表示在目标方法执行前来实施增强
  2. 后置增强 (org.springframework.aop.AfterReturningAdvice)

    表示在目标方法执行后来实施增强

  3. 环绕增强 (org.aopalliance.intercept.MethodInterceptor)

    表示在目标方法执行前后同时实施增强

  4. 异常抛出增强 (org.springframework.aop.ThrowsAdvice) 表示在目标方法抛出异常后来实施增强
  5. 引介增强 (org.springframework.aop.introductioninterceptor)

    表示在目标类中添加一些新的方法和属性

其中引介增强是一种特殊的增强。它可以在目标类中添加属性和方法,通过拦截定义一个接口,让目标代理实现这个接口。

  • 引介增强的连接点是类级别的,而前面的几种则是方法级别的。
  • 环绕增强是AOP联盟定义的接口,其他四种增强接口则是Spring定义的接口。

这些增强接口都有一些方法,通过实现这些接口方法,并在接口方法中定义横切逻辑,就可以将它们织入目标类方法的相应连接点位置。

用Spring实现的话: 通过实现这些增强接口,在实现这些接口的方法当中定义横切逻辑,然后通过配置Spring的配置文件就可以完成将增强织入到目标方法当中了。

前置增强

在Spring当中,仅支持方法级别的增强,利用MethodBeforeAdvice实现,表示在目标方法执行前实施增强

实例一:通过代码实现增强

代码已托管到Github—> https://github.com/yangshangwei/SpringMaster

示例演示: 对服务生的服务用语进行强制规范。

我们假设服务生只需要干两件事情:

  • 1.欢迎顾客
  • 2.对顾客提供服务
Spring-AOP 增强(Advice)5种类型和创建增强类概述增强类型提示

代码说明:

创建业务接口类:Waiter.java

创建业务实现类:NativeWaiter.java

创建业务增强类:GreetingBeforeAdvice.java

创建增强测试类:BeforceAdviceTest.java

业务接口类

package com.xgj.aop.spring.advice.beforceAdvice;

public interface Waiter {

	void greetTo(String name);

	void serverTo(String name);
}           

复制

业务实现类

package com.xgj.aop.spring.advice.beforceAdvice;

public class NaiveWaiter implements Waiter {

	@Override
	public void greetTo(String name) {
		System.out.println("greetTo " + name + "\n");
	}

	@Override
	public void serverTo(String name) {
		System.out.println("serverTo " + name);
	}

}           

复制

业务增强类

package com.xgj.aop.spring.advice.beforceAdvice;

import java.lang.reflect.Method;

import org.springframework.aop.MethodBeforeAdvice;

public class GreetingBeforeAdvice implements MethodBeforeAdvice {

	/**
	 * 前置增强方法 当该方法发生异常时,将阻止目标方法的执行
	 * 
	 * @param method
	 *            目标类方法
	 * @param objects
	 *            目标类方法入参
	 * @param target
	 *            目标类对象实例
	 * @throws Throwable
	 */
	@Override
	public void before(Method method, Object[] args, Object target)
			throws Throwable {
		String clientName = (String) args[0];
		System.out.println("How are you " + clientName + " ?");
	}
}           

复制

说明:

BeforeAdvice是前置增强的接口,方法前置增强的MethodBeforeAdvice接口是其子类。 Spring目前仅提供方法调用的前置增强。

MethodBeforeAdvice接口定义了唯一的方法

void before(Method method, Object[] args, Object target) throws Throwable;           

复制

Spring-AOP 增强(Advice)5种类型和创建增强类概述增强类型提示

参数解释:

  • method:目标类的方法
  • args:目标类方法的入参
  • target:目标类实例

当该方法发生异常时,将阻止目标类方法的执行

业务测试类

package com.xgj.aop.spring.advice.beforceAdvice;

import org.junit.Test;
import org.springframework.aop.BeforeAdvice;
import org.springframework.aop.framework.ProxyFactory;

public class BeforceAdviceTest {
	@Test
	public void test() {
		// 创建目标对象
		Waiter target = new NaiveWaiter();
		// 创建增强类对象
		BeforeAdvice advice = new GreetingBeforeAdvice();

		// (1)创建代理工厂对象 - Spring提供的代理工厂
		ProxyFactory factory = new ProxyFactory();

		// (2)设置代理目标
		factory.setTarget(target);

		// (3)为代理目标添加增强
		factory.addAdvice(advice);

		// (4)获取代理类
		Waiter proxy = (Waiter) factory.getProxy();

		// 调用目标类方法
		proxy.greetTo("Xiao");
		proxy.serverTo("GongJiang");
	}
}           

复制

运行单元测试:

Spring-AOP 增强(Advice)5种类型和创建增强类概述增强类型提示

实例二 通过配置文件实现前置增强-ProxyFactory

我们上面的前置增强是使用org.springframework.aop.framework.ProxyFactory通过编码的方式将GreetingBeforeAdvice的增强织入目标类NaiveWaiter中,那如何通过Spring配置文件的方式实现同样的效果呢?

请参考另外一篇博文 Spring-AOP 通过配置文件实现前置增强-ProxyFactory

后置增强

请参阅另外一篇博客 Spring-AOP 通过配置文件实现 后置增强

环绕增强

请参阅另外一篇博客 Spring-AOP 通过配置文件实现 环绕增强

异常抛出增强

请参阅另外一篇博客 Spring-AOP 通过配置文件实现 异常抛出增强

引介增强

请参阅另外一篇博客 Spring-AOP 通过配置文件实现 引介增强

提示

在Spring4.0中,基于CGLib的类代理不再要求目标类必须有无参构造函数。这样在使用CGLib类时,不再需要特别关注目标类是否有无参构造函数。 这样的话,增强的目标Bean就可以使用构造函数注入了。