浅谈spring aop的五种通知类型

2025-05-29 0 103

spring aop通知(advice)分成五类:

前置通知[before advice]:在连接点前面执行,前置通知不会影响连接点的执行,除非此处抛出异常。

正常返回通知[after returning advice]:在连接点正常执行完成后执行,如果连接点抛出异常,则不会执行。

异常返回通知[after throwing advice]:在连接点抛出异常后执行。

返回通知[after (finally) advice]:在连接点执行完成后执行,不管是正常执行完成,还是抛出异常,都会执行返回通知中的内容。

环绕通知[around advice]:环绕通知围绕在连接点前后,比如一个方法调用的前后。这是最强大的通知类型,能在方法调用前后自定义一些操作。

环绕通知还需要负责决定是继续处理join point(调用proceedingjoinpoint的proceed方法)还是中断执行。
接下来通过编写示例程序来测试一下五种通知类型:

定义接口

?

1

2

3

4

5

6

7

8

9

10

11

12

13
package com.chenqa.springaop.example.service;

public interface bankservice {

/**

* 模拟的银行转账

* @param from 出账人

* @param to 入账人

* @param account 转账金额

* @return

*/

public boolean transfer(string form, string to, double account);

}

编写实现类

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15
package com.chenqa.springaop.example.service.impl;

import com.chenqa.springaop.example.service.bankservice;

public class bcmbankserviceimpl implements bankservice {

public boolean transfer(string form, string to, double account) {

if(account<100) {

throw new illegalargumentexception("最低转账金额不能低于100元");

}

system.out.println(form+"向"+to+"交行账户转账"+account+"元");

return false;

}

}

修改spring配置文件,添加以下内容:

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15
<!-- bankservice bean -->

<bean id="bankservice" class="com.chenqa.springaop.example.service.impl.bcmbankserviceimpl"/>

<!-- 切面 -->

<bean id="myaspect" class="com.chenqa.springaop.example.aspect.myaspect"/>

<!-- aop配置 -->

<aop:config>

<aop:aspect ref="myaspect">

<aop:pointcut expression="execution(* com.chenqa.springaop.example.service.impl.*.*(..))" id="pointcut"/>

<aop:before method="before" pointcut-ref="pointcut"/>

<aop:after method="after" pointcut-ref="pointcut"/>

<aop:after-returning method="afterreturning" pointcut-ref="pointcut"/>

<aop:after-throwing method="afterthrowing" pointcut-ref="pointcut"/>

<aop:around method="around" pointcut-ref="pointcut"/>

</aop:aspect>

</aop:config>

编写测试程序

?

1

2

3
applicationcontext context = new classpathxmlapplicationcontext("spring-aop.xml");

bankservice bankservice = context.getbean("bankservice", bankservice.class);

bankservice.transfer("张三", "李四", 200);

执行后输出:

浅谈spring aop的五种通知类型

将测试程序中的200改成50,再执行后输出:

浅谈spring aop的五种通知类型

通过测试结果可以看出,五种通知的执行顺序为:

前置通知→环绕通知→正常返回通知/异常返回通知→返回通知,可以多次执行来查看。

情况一: 一个方法只被一个aspect类拦截

当一个方法只被一个aspect拦截时,这个aspect中的不同advice是按照怎样的顺序进行执行的呢?请看:

添加pointcut类

该pointcut用来拦截test包下的所有类中的所有方法。

?

1

2

3

4

5

6

7

8

9

10
package test;

import org.aspectj.lang.annotation.pointcut;

public class pointcuts {

@pointcut(value = "within(test.*)")

public void aopdemo() {

}

}

添加aspect类

该类中的advice将会用到上面的pointcut,使用方法请看各个advice的value属性。

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38
package test;

import org.aspectj.lang.joinpoint;

import org.aspectj.lang.proceedingjoinpoint;

import org.aspectj.lang.annotation.*;

import org.springframework.stereotype.component;

@component

@aspect

public class aspect1 {

@before(value = "test.pointcuts.aopdemo()")

public void before(joinpoint joinpoint) {

system.out.println("[aspect1] before advise");

}

@around(value = "test.pointcuts.aopdemo()")

public void around(proceedingjoinpoint pjp) throws throwable{

system.out.println("[aspect1] around advise 1");

pjp.proceed();

system.out.println("[aspect1] around advise2");

}

@afterreturning(value = "test.pointcuts.aopdemo()")

public void afterreturning(joinpoint joinpoint) {

system.out.println("[aspect1] afterreturning advise");

}

@afterthrowing(value = "test.pointcuts.aopdemo()")

public void afterthrowing(joinpoint joinpoint) {

system.out.println("[aspect1] afterthrowing advise");

}

@after(value = "test.pointcuts.aopdemo()")

public void after(joinpoint joinpoint) {

system.out.println("[aspect1] after advise");

}

}

添加测试用controller

添加一个用于测试的controller,这个controller中只有一个方法,但是它会根据参数值的不同,会作出不同的处理:一种是正常返回一个对象,一种是抛出异常(因为我们要测试@afterthrowing这个advice)

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48
package test;

import test.exception.testexception;

import org.springframework.http.httpstatus;

import org.springframework.web.bind.annotation.*;

@restcontroller

@requestmapping(value = "/aop")

public class aoptestcontroller {

@responsestatus(httpstatus.ok)

@requestmapping(value = "/test", method = requestmethod.get)

public result test(@requestparam boolean throwexception) {

// case 1

if (throwexception) {

system.out.println("throw an exception");

throw new testexception("mock a server exception");

}

// case 2

system.out.println("test ok");

return new result() {{

this.setid(111);

this.setname("mock a result");

}};

}

public static class result {

private int id;

private string name;

public int getid() {

return id;

}

public void setid(int id) {

this.id = id;

}

public string getname() {

return name;

}

public void setname(string name) {

this.name = name;

}

}

}

测试 正常情况

在浏览器直接输入以下的url,回车:http://192.168.142.8:7070/aoptest/v1/aop/test?throwexception=false1

我们会看到输出的结果是:

?

1

2

3

4

5

6
[aspect1] around advise 1

[aspect1] before advise

test ok

[aspect1] around advise2

[aspect1] after advise

[aspect1] afterreturning advise

测试 异常情况

在浏览器中直接输入以下的url,回车:http://192.168.142.8:7070/aoptest/v1/aop/test?throwexception=true1

我们会看到输出的结果是:

?

1

2

3

4

5
[aspect1] around advise 1

[aspect1] before advise

throw an exception

[aspect1] after advise

[aspect1] afterthrowing advise

结论

在一个方法只被一个aspect类拦截时,aspect类内部的 advice 将按照以下的顺序进行执行:

正常情况:

浅谈spring aop的五种通知类型

异常情况:

浅谈spring aop的五种通知类型

情况二: 同一个方法被多个aspect类拦截

此处举例为被两个aspect类拦截。

有些情况下,对于两个不同的aspect类,不管它们的advice使用的是同一个pointcut,还是不同的pointcut,都有可能导致同一个方法被多个aspect类拦截。那么,在这种情况下,这多个aspect类中的advice又是按照怎样的顺序进行执行的呢?请看:

pointcut类保持不变

添加一个新的aspect类

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38
package test;

import org.aspectj.lang.joinpoint;

import org.aspectj.lang.proceedingjoinpoint;

import org.aspectj.lang.annotation.*;

import org.springframework.stereotype.component;

@component

@aspect

public class aspect2 {

@before(value = "test.pointcuts.aopdemo()")

public void before(joinpoint joinpoint) {

system.out.println("[aspect2] before advise");

}

@around(value = "test.pointcuts.aopdemo()")

public void around(proceedingjoinpoint pjp) throws throwable{

system.out.println("[aspect2] around advise 1");

pjp.proceed();

system.out.println("[aspect2] around advise2");

}

@afterreturning(value = "test.pointcuts.aopdemo()")

public void afterreturning(joinpoint joinpoint) {

system.out.println("[aspect2] afterreturning advise");

}

@afterthrowing(value = "test.pointcuts.aopdemo()")

public void afterthrowing(joinpoint joinpoint) {

system.out.println("[aspect2] afterthrowing advise");

}

@after(value = "test.pointcuts.aopdemo()")

public void after(joinpoint joinpoint) {

system.out.println("[aspect2] after advise");

}

}

测试用controller也不变

还是使用上面的那个controller。但是现在 aspect1 和 aspect2 都会拦截该controller中的方法。

下面继续进行测试!

测试 正常情况

在浏览器直接输入以下的url,回车:http://192.168.142.8:7070/aoptest/v1/aop/test?throwexception=false1

我们会看到输出的结果是:

?

1

2

3

4

5

6

7

8

9

10

11
[aspect2] around advise 1

[aspect2] before advise

[aspect1] around advise 1

[aspect1] before advise

test ok

[aspect1] around advise2

[aspect1] after advise

[aspect1] afterreturning advise

[aspect2] around advise2

[aspect2] after advise

[aspect2] afterreturning advise

但是这个时候,我不能下定论说 aspect2 肯定就比 aspect1 先执行。

不信?你把服务务器重新启动一下,再试试,说不定你就会看到如下的执行结果:

?

1

2

3

4

5

6

7

8

9

10

11
[aspect1] around advise 1

[aspect1] before advise

[aspect2] around advise 1

[aspect2] before advise

test ok

[aspect2] around advise2

[aspect2] after advise

[aspect2] afterreturning advise

[aspect1] around advise2

[aspect1] after advise

[aspect1] afterreturning advise

也就是说,这种情况下, aspect1 和 aspect2 的执行顺序是未知的。那怎么解决呢?不急,下面会给出解决方案。

测试 异常情况

在浏览器中直接输入以下的url,回车:http://192.168.142.8:7070/aoptest/v1/aop/test?throwexception=true1

我们会看到输出的结果是:

?

1

2

3

4

5

6

7

8

9
[aspect2] around advise 1

[aspect2] before advise

[aspect1] around advise 1

[aspect1] before advise

throw an exception

[aspect1] after advise

[aspect1] afterthrowing advise

[aspect2] after advise

[aspect2] afterthrowing advise

同样地,如果把服务器重启,然后再测试的话,就可能会看到如下的结果:

?

1

2

3

4

5

6

7

8

9
[aspect1] around advise 1

[aspect1] before advise

[aspect2] around advise 1

[aspect2] before advise

throw an exception

[aspect2] after advise

[aspect2] afterthrowing advise

[aspect1] after advise

[aspect1] afterthrowing advise

也就是说,同样地,异常情况下, aspect1 和 aspect2 的执行顺序也是未定的。

那么在 情况二 下,如何指定每个 aspect 的执行顺序呢?

方法有两种:

  1. 实现org.springframework.core.ordered接口,实现它的getorder()方法
  2. 给aspect添加@order注解,该注解全称为:org.springframework.core.annotation.order

不管采用上面的哪种方法,都是值越小的 aspect 越先执行。

比如,我们为 apsect1 和 aspect2 分别添加 @order 注解,如下:

?

1

2

3

4

5

6

7

8

9

10

11

12

13
@order(5)

@component

@aspect

public class aspect1 {

// ...

}

@order(6)

@component

@aspect

public class aspect2 {

// ...

}

这样修改之后,可保证不管在任何情况下, aspect1 中的 advice 总是比 aspect2 中的 advice 先执行。如下图所示:

浅谈spring aop的五种通知类型

注意点

如果在同一个 aspect 类中,针对同一个 pointcut,定义了两个相同的 advice(比如,定义了两个 @before),那么这两个 advice 的执行顺序是无法确定的,哪怕你给这两个 advice 添加了 @order 这个注解,也不行。这点切记。

对于@around这个advice,不管它有没有返回值,但是必须要方法内部,调用一下pjp.proceed();否则,controller 中的接口将没有机会被执行,从而也导致了@before这个advice不会被触发。比如,我们假设正常情况下,执行顺序为”aspect2 -> apsect1 -> controller”,如果,我们把aspect1中的@around中的pjp.proceed();给删掉,那么,我们看到的输出结果将是:

?

1

2

3

4

5

6

7

8

9
[aspect2] around advise 1

[aspect2] before advise

[aspect1] around advise 1

[aspect1] around advise2

[aspect1] after advise

[aspect1] afterreturning advise

[aspect2] around advise2

[aspect2] after advise

[aspect2] afterreturning advise

从结果可以发现, controller 中的 接口 未被执行,aspect1 中的@beforeadvice 也未被执行。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持快网idc。

原文链接:http://blog.csdn.net/qq_35873847/article/details/78624941

收藏 (0) 打赏

感谢您的支持,我会继续努力的!

打开微信/支付宝扫一扫,即可进行扫码打赏哦,分享从这里开始,精彩与您同在
点赞 (0)

声明:本站所有文章,如无特殊说明或标注,均为本站原创发布。任何个人或组织,在未征得本站同意时,禁止复制、盗用、采集、发布本站内容到任何网站、书籍等各类媒体平台。如若本站内容侵犯了原著者的合法权益,可联系我们进行处理。

快网idc优惠网 建站教程 浅谈spring aop的五种通知类型 https://www.kuaiidc.com/113674.html

相关文章

发表评论
暂无评论