Spring AOP通知实例 – Advice

本文内容纲要:

- 

- Spring AOP 通知

- 1. 之前通知(Before Advice)

-

-

-

- 总结

Spring AOP(面向方面编程)框架,用于在模块化方面的横切关注点。简单得说,它只是一个拦截器拦截一些过程,例如,当一个方法执行,Spring AOP 可以劫持一个执行的方法,在方法执行之前或之后添加额外的功能。

在Spring AOP中,有 4 种类型通知(advices)的支持:

  • 通知(Advice)之前 - 该方法执行前运行
  • 通知(Advice)返回之后 – 运行后,该方法返回一个结果
  • 通知(Advice)抛出之后 – 运行方法抛出异常后,
  • 环绕通知 – 环绕方法执行运行,结合以上这三个通知。

下面的例子显示Spring AOP 通知如何工作。

简单的 Spring 例子

创建一个简单的客户服务类及一个print方法作为演示。

public class CustomerService {

private String name;

private String url;

public void setName(String name) {

this.name = name;

}

public void setUrl(String url) {

this.url = url;

}

public void printName() {

System.out.println("Customer name : " + this.name);

}

public void printURL() {

System.out.println("Customer website : " + this.url);

}

public void printThrowException() {

throw new IllegalArgumentException();

}

}

File : beans.xml – 一个bean配置文件

<?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">

<bean id="customerService" class="com.ray.customer.service.CustomerService">

<property name="name" value="Ray"/>

<property name="url" value="http://www.baidu.com"/>

</bean>

</beans>

执行

public class Test {

public static void main(String[] args) {

ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");

CustomerService cust = (CustomerService) context.getBean("customerService");

System.out.println("*************************");

cust.printName();

System.out.println("*************************");

cust.printURL();

System.out.println("*************************");

try {

cust.printThrowException();

}catch(Exception e) {

}

}

}

输出

*************************

Customer name : Ray

*************************

Customer website : http://www.baidu.com

*************************

一个简单的Spring项目用来注入(DI)bean和输出一些字符串。

Spring AOP 通知

现在,附加 Spring AOP 建议到上述的客户服务。

1. 之前通知(Before Advice)

它会在方法执行之前执行。创建一个实现 MethodBeforeAdvice 接口的类。

public class BeforeMethod implements MethodBeforeAdvice {

@Override

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

System.out.println("BeforeMethod : Before!");

}

}

在 bean 配置文件(beans.xml),创建一个 bean 的 BeforeMethod 类,并命名为“customerServiceProxy” 作为一个新的代理对象。

  • ‘target’ – 定义你想拦截的bean。

  • ‘interceptorNames’ – 定义要应用这个代理/目标对象的类(通知)。

    <bean id="customerService" class="com.ray.customer.service.CustomerService">

    <property name="name" value="Ray"/>

    <property name="url" value="http://www.baidu.com"/>

    </bean>

    <bean id="BeforeMethodBean" class="com.ray.aop.BeforeMethod"/>

    <bean id="customerServiceProxy" class="org.springframework.aop.framework.ProxyFactoryBean">

    <property name="target" ref="customerService"/>

    <property name="interceptorNames">

    <list>

    <value>BeforeMethodBean</value>

    </list>

    </property>

    </bean>

再次运行它,现在得到新的 customerServiceProxy bean,而不是原来的CustomerService bean。

public class Test {

public static void main(String[] args) {

ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");

CustomerService cust = (CustomerService) context.getBean("customerServiceProxy");

System.out.println("*************************");

cust.printName();

System.out.println("*************************");

cust.printURL();

System.out.println("*************************");

try {

cust.printThrowException();

}catch(Exception e) {

}

}

}

输出结果

*************************

BeforeMethod : Before method!

Customer name : Ray

*************************

BeforeMethod : Before method!

Customer website : http://www.baidu.com

*************************

BeforeMethod : Before method!

它将运行 BeforeMethod 的 before() 方法,在每个 CustomerService 的方法之前执行。

2.返回后通知(After Returning Advice)

该方法返回一个结果之后它将执行。创建一个实现AfterReturningAdvice接口的类。

public class AfterMethod implements AfterReturningAdvice {

@Override

public void afterReturning(Object returnValue, Method method, Object[] args, Object target) throws Throwable {

System.out.println("HijackAfterMethod : After method hijacked!");

}

}

bean配置文件

<?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">

<bean id="customerService" class="com.ray.customer.service.CustomerService">

<property name="name" value="Ray"/>

<property name="url" value="http://www.baidu.com"/>

</bean>

<bean id="AfterMethodBean" class="com.ray.aop.AfterMethod"/>

<bean id="customerServiceProxy" class="org.springframework.aop.framework.ProxyFactoryBean">

<property name="target" ref="customerService"/>

<property name="interceptorNames">

<list>

<value>hAfterMethodBean</value>

</list>

</property>

</bean>

</beans>

再次运行,输出结果

*************************

Customer name : Ray

AfterMethod : After method!

*************************

Customer website : http://www.baidu.com

AfterMethod : After method!

*************************

它将运行 AfterMethod 的 afterReturning()方法,在每次 CustomerService 方法返回结果之后。

可以看到输出结果,每一个customerService的method运行返回结果后,都将再执行AfterMethod的afterReturning方法。但是执行到cust.printThrowException()后,直接抛出异常,方法没有正常执行完毕(或者说没有返回结果),所以不运行切入的afterReturning方法。

3.抛出后通知(Afetr Throwing Advice)

它将在执行方法抛出一个异常后。创建一个实现ThrowsAdvice接口的类,并创建一个afterThrowing方法拦截抛出:IllegalArgumentException异常。

public class ThrowException implements ThrowsAdvice {

public void afterThrowing(IllegalArgumentException e)throws Throwable{

System.out.println("ThrowException : Throw exception!");

}

}

bean配置文件

<?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">

<bean id="customerService" class="com.ray.customer.service.CustomerService">

<property name="name" value="Ray"/>

<property name="url" value="http://www.baidu.com"/>

</bean>

<bean id="ThrowExceptionBean" class="com.ray.aop.ThrowException"/>

<bean id="customerServiceProxy" class="org.springframework.aop.framework.ProxyFactoryBean">

<property name="target" ref="customerService"/>

<property name="interceptorNames">

<list>

<value>ThrowExceptionBean</value>

</list>

</property>

</bean>

</beans>

再次运行,输出结果

*************************

Customer name : Ray

*************************

Customer website : http://www.baidu.com

*************************

ThrowException : Throw exception!

它将运行 ThrowException 的 afterThrowing()方法,如果 CustomerService 的方法抛出异常。

当运行CustomerService中的printThrowException方法时,认为的抛出IllegalArgumentException异常,被ThrowException截获,运行其中的afterThrowing方法。注意,如果抛出异常不是IllegalArgumentException,则不能被截获。

4.环绕通知(Around Advice)

它结合了上面的三个通知,在方法执行过程中执行。创建一个实现了MethodInterceptor接口的类。必须调用“methodInvocation.proceed();” 继续在原来的方法执行,否则原来的方法将不会执行。

public class AroundMethod implements MethodInterceptor {

@Override

public Object invoke(MethodInvocation methodInvocation) throws Throwable {

System.out.println("Method name:" + methodInvocation.getMethod().getName());

System.out.println("Method arguments:" + Arrays.toString(methodInvocation.getArguments()));

//same with MethodBeforeAdvice

System.out.println("AroundMethod : Before method!");

try {

//proceed to original method call

Object result = methodInvocation.proceed();

//same with AfterReturningAdvice

System.out.println("AroundMethod : Before after!");

return result;

}catch(IllegalArgumentException e) {

//same with ThrowsAdvice

System.out.println("AroundMethod : Throw exception!");

throw e;

}

}

}

bean配置文件

<?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">

<bean id="customerService" class="com.ray.customer.service.CustomerService">

<property name="name" value="Ray"/>

<property name="url" value="http://www.baidu.com"/>

</bean>

<bean id="AroundMethodBean" class="com.ray.aop.AroundMethod"/>

<bean id="customerServiceProxy" class="org.springframework.aop.framework.ProxyFactoryBean">

<property name="target" ref="customerService"/>

<property name="interceptorNames">

<list>

<value>AroundMethodBean</value>

</list>

</property>

</bean>

</beans>

再次运行,输出结果

*************************

Method name:printName

Method arguments:[]

AroundMethod : Before method!

Customer name : Ray

AroundMethod : Before after!

*************************

Method name:printURL

Method arguments:[]

AroundMethod : Before method!

Customer website : http://www.baidu.com

AroundMethod : Before after!

*************************

Method name:printThrowException

Method arguments:[]

AroundMethod : Before method!

AroundMethod : Throw exception!

它将运行AroundMethod 的 invoke()方法,在每一个 CustomerService 方法执行后。

总结

大部分的 Spring 开发者都只是实现了“环绕通知”,因为它可以对所有通知类型,但更好的做法应该是选择最合适的通知类型来满足要求。

切入点

在这个例子中,在一客户服务类中的所有方法都自动拦截(通知)。但在大多数情况下,可能需要使用切入点和Advisor通过它的方法名拦截它的方法。

本文内容总结:,Spring AOP 通知,1. 之前通知(Before Advice),,,,总结,

原文链接:https://www.cnblogs.com/Remenber-Ray/p/12320160.html

以上是 Spring AOP通知实例 – Advice 的全部内容, 来源链接: utcz.com/z/362896.html

回到顶部