前言
在之前的一篇文章中我们讲到spring的xml实现,这里我们讲讲使用注解如何实现aop呢。前面已经讲过aop的简单理解了,这里就不在赘述了。话不多说,来一起看看详细的介绍:
注解方式实现aop我们主要分为如下几个步骤:
1.在切面类(为切点服务的类)前用@Aspect注释修饰,声明为一个切面类。
2.用@Pointcut注释声明一个切点,目的是为了告诉切面,谁是它的服务对象。(此注释修饰的方法的方法体为空,不需要写功能比如 public void say(){};
就可以了,方法名可以被候命的具体服务功能所以引用,它可以被理解为切点对象的一个代理对象方法)
3.在对应的方法前用对应的通知类型注释修饰,将对应的方法声明称一个切面功能,为了切点而服务
4.在spring配置文件中开启aop注释自动代理。如: <aop:aspectj-autoproxy/>
这样讲可能还是很抽象,那么,废话不多说,我们代码说话,代码如下:
骑士类:(看过上一篇内容的就知道骑士是什么东西了,嘿嘿嘿)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
|
package com.cjh.aop2;
import org.springframework.stereotype.Component;
/**
* @author Caijh
*
* 2017年7月11日 下午3:53:19
*/
@Component ( "knight" )
public class BraveKnight {
public void saying(){
System.out.println( "我是骑士..(切点方法)" );
}
}
|
切面类:(注释主要在这里体现)
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
49
50
51
52
53
54
55
56
57
58
59
|
package com.cjh.aop2;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.After;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;
import org.springframework.stereotype.Component;
/**
* @author Caijh
* email:codecjh@163.com
* 2017年7月12日 上午9:31:43
*/
/**
* 注解方式声明aop
* 1.用@Aspect注解将类声明为切面(如果用@Component("")注解注释为一个bean对象,那么就要在spring配置文件中开启注解扫描,<context:component-scan base-package="com.cjh.aop2"/>
* 否则要在spring配置文件中声明一个bean对象)
* 2.在切面需要实现相应方法的前面加上相应的注释,也就是通知类型。
* 3.此处有环绕通知,环绕通知方法一定要有ProceedingJoinPoint类型的参数传入,然后执行对应的proceed()方法,环绕才能实现。
*/
@Component ( "annotationTest" )
@Aspect
public class AnnotationTest {
//定义切点
@Pointcut ( "execution(* *.saying(..))" )
public void sayings(){}
/**
* 前置通知(注解中的sayings()方法,其实就是上面定义pointcut切点注解所修饰的方法名,那只是个代理对象,不需要写具体方法,
* 相当于xml声明切面的id名,如下,相当于id="embark",用于供其他通知类型引用)
* <aop:config>
<aop:aspect ref="mistrel">
<!-- 定义切点 -->
<aop:pointcut expression="execution(* *.saying(..))" id="embark"/>
<!-- 声明前置通知 (在切点方法被执行前调用) -->
<aop:before method="beforSay" pointcut-ref="embark"/>
<!-- 声明后置通知 (在切点方法被执行后调用) -->
<aop:after method="afterSay" pointcut-ref="embark"/>
</aop:aspect>
</aop:config>
*/
@Before ( "sayings()" )
public void sayHello(){
System.out.println( "注解类型前置通知" );
}
//后置通知
@After ( "sayings()" )
public void sayGoodbey(){
System.out.println( "注解类型后置通知" );
}
//环绕通知。注意要有ProceedingJoinPoint参数传入。
@Around ( "sayings()" )
public void sayAround(ProceedingJoinPoint pjp) throws Throwable{
System.out.println( "注解类型环绕通知..环绕前" );
pjp.proceed(); //执行方法
System.out.println( "注解类型环绕通知..环绕后" );
}
}
|
spring配置文件:
1
2
3
4
5
6
7
8
9
10
11
12
13
|
<? xml version = "1.0" encoding = "UTF-8" ?>
< beans xmlns = "http://www.springframework.org/schema/beans"
xmlns:xsi = "http://www.w3.org/2001/XMLSchema-instance"
xmlns:context = "http://www.springframework.org/schema/context"
xmlns:aop = "http://www.springframework.org/schema/aop"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-4.3.xsd
http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.3.xsd">
<!-- 开启注解扫描 -->
< context:component-scan base-package = "com.cjh.aop2" />
<!-- 开启aop注解方式,此步骤s不能少,这样java类中的aop注解才会生效 -->
< aop:aspectj-autoproxy />
</ beans >
|
测试代码:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
|
package com.cjh.aop2;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
/**
*
* @author Caijh
* email:codecjh@163.com
* 2017年7月11日 下午6:27:06
*/
public class Test {
public static void main(String[] args) {
ApplicationContext ac = new ClassPathXmlApplicationContext( "com/cjh/aop2/beans.xml" );
BraveKnight br = (BraveKnight) ac.getBean( "knight" );
br.saying();
}
}
|
运行结果:
注解类型环绕通知..环绕前
注解类型前置通知
我是骑士..(切点方法)
注解类型环绕通知..环绕后
注解类型后置通知
========================分割线===================================
因为使用了注解方式,所以配置文件少了很多内容,只需要一句<context:component-scan base-package="com.cjh.aop2"/>
声明要扫描的包,框架会自动扫描注释并生成bean对象。有个@Component("knight")
这个注释,和<bean id="knight" class="com.cjh.aop2.BraveKnight"/>
这个配置时一样的意思,框架会自动识别并创建名为knight的BraveKnight对象。所以有了注释,只需要开启注释扫描配置就好了,无需再做相同的bean配置。
如果运行过程中出现Spring aop : error at ::0 can't find referenced pointcut sleepPonit
的错误,那么很可能是spring的包的版本问题,
我用的是spring4的版本,然后还需要加aspectjrt-1.7.4.jar和aspectjweaver-1.7.4.jar两个包
下载地址:aspectjrt_aspectjweaver.rar
工程目录如下:(打码码的类不用管,只是为了不为大家造成误导,所以划掉了)
通知注解类型如下:
总结
以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作能带来一定的帮助,如果有疑问大家可以留言交流,谢谢大家对服务器之家的支持
原文链接:http://www.cnblogs.com/caijh/p/7154691.html