Spring 中aop切面注解实现 spring中aop的注解实现方式简单实例

spring中aop的注解实现方式简单实例

 

  上篇中我们讲到spring的xml实现,这里我们讲讲使用注解如何实现aop呢。前面已经讲过aop的简单理解了,这里就不在赘述了。

注解方式实现aop我们主要分为如下几个步骤(自己整理的,有更好的方法的话,欢迎交流[email protected]):

  1.在切面类(为切点服务的类)前用@Aspect注释修饰,声明为一个切面类。

  2.用@Pointcut注释声明一个切点,目的是为了告诉切面,谁是它的服务对象。(此注释修饰的方法的方法体为空,不需要写功能比如 public void say(){};就可以了,方法名可以被候命的具体服务功能所以引用,它可以被理解为切点对象的一个代理对象方法)

  3.在对应的方法前用对应的通知类型注释修饰,将对应的方法声明称一个切面功能,为了切点而服务

  4.在spring配置文件中开启aop注释自动代理。如:<aop:aspectj-autoproxy/>

这样讲可能还是很抽象,那么,废话不多说,我们代码说话,代码如下:

骑士类:(看过上一篇内容的就知道骑士是什么东西了,嘿嘿嘿)

复制代码
 1 package com.cjh.aop2;
 2 
 3 import org.springframework.stereotype.Component;
 4 
 5 /**
 6  * @author 古斌
 7  *
 9  */
10 @Component("knight")
11 public class BraveKnight {
12     public void saying(){
13         System.out.println("我是骑士..(切点方法)");
14     }
15 }
复制代码

切面类:(注释主要在这里体现)

复制代码
 1 package com.cjh.aop2;
 2 
 3 import org.aspectj.lang.ProceedingJoinPoint;
 4 import org.aspectj.lang.annotation.After;
 5 import org.aspectj.lang.annotation.Around;
 6 import org.aspectj.lang.annotation.Aspect;
 7 import org.aspectj.lang.annotation.Before;
 8 import org.aspectj.lang.annotation.Pointcut;
 9 import org.springframework.stereotype.Component;
10 
11 /**
15  */
16 /**
17  * 注解方式声明aop
18  * 1.用@Aspect注解将类声明为切面(如果用@Component("")注解注释为一个bean对象,那么就要在spring配置文件中开启注解扫描,<context:component-scan base-package="com.cjh.aop2"/>
19  *      否则要在spring配置文件中声明一个bean对象)
20  * 2.在切面需要实现相应方法的前面加上相应的注释,也就是通知类型。
21  * 3.此处有环绕通知,环绕通知方法一定要有ProceedingJoinPoint类型的参数传入,然后执行对应的proceed()方法,环绕才能实现。
22  */
23 @Component("annotationTest")
24 @Aspect
25 public class AnnotationTest {
26     //定义切点
27     @Pointcut("execution(* *.saying(..))")
28     public void sayings(){}
29     /**
30      * 前置通知(注解中的sayings()方法,其实就是上面定义pointcut切点注解所修饰的方法名,那只是个代理对象,不需要写具体方法,
31      * 相当于xml声明切面的id名,如下,相当于id="embark",用于供其他通知类型引用)
32      * <aop:config>
33         <aop:aspect ref="mistrel">
34             <!-- 定义切点 -->
35             <aop:pointcut expression="execution(* *.saying(..))" id="embark"/>
36             <!-- 声明前置通知 (在切点方法被执行前调用) -->
37             <aop:before method="beforSay" pointcut-ref="embark"/>
38             <!-- 声明后置通知 (在切点方法被执行后调用) -->
39             <aop:after method="afterSay" pointcut-ref="embark"/>
40         </aop:aspect>
41        </aop:config>
42      */
43     @Before("sayings()")
44     public void sayHello(){
45         System.out.println("注解类型前置通知");
46     }
47     //后置通知
48     @After("sayings()")
49     public void sayGoodbey(){
50         System.out.println("注解类型后置通知");
51     }
52     //环绕通知。注意要有ProceedingJoinPoint参数传入。
53     @Around("sayings()")
54     public void sayAround(ProceedingJoinPoint pjp) throws Throwable{
55         System.out.println("注解类型环绕通知..环绕前");
56         pjp.proceed();//执行方法
57         System.out.println("注解类型环绕通知..环绕后");
58     }
59 }
复制代码

spring配置文件:

复制代码
 1 <?xml version="1.0" encoding="UTF-8"?>
 2 <beans xmlns="http://www.springframework.org/schema/beans"
 3     xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
 4     xmlns:context="http://www.springframework.org/schema/context"
 5     xmlns:aop="http://www.springframework.org/schema/aop"
 6     xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
 7         http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-4.3.xsd
 8         http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.3.xsd">
 9     <!-- 开启注解扫描 -->
10     <context:component-scan base-package="com.cjh.aop2"/>
11     <!-- 开启aop注解方式,此步骤s不能少,这样java类中的aop注解才会生效 -->
12     <aop:aspectj-autoproxy/>
13 </beans>
复制代码

测试代码:

复制代码
 1 package com.cjh.aop2;
 2 
 3 import org.springframework.context.ApplicationContext;
 4 import org.springframework.context.support.ClassPathXmlApplicationContext;
 5 
 6 /**
 7  * 
11  */
12 public class Test {
13     public static void main(String[] args) {
14         ApplicationContext ac = new ClassPathXmlApplicationContext("com/cjh/aop2/beans.xml");
15         BraveKnight br = (BraveKnight) ac.getBean("knight");
16         br.saying();
17     }
18 }
复制代码

运行结果:

注解类型环绕通知..环绕前
注解类型前置通知
我是骑士..(切点方法)
注解类型环绕通知..环绕后
注解类型后置通知

========================分割线===================================

  因为使用了注解方式,所以配置文件少了很多内容,只需要一句<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两个包,云盘地址:链接:http://pan.baidu.com/s/1qXQurO4 密码:nc4i

工程目录如下:(打码码的类不用管,只是为了不为大家造成误导,所以划掉了)

通知注解类型如下:

  上篇中我们讲到spring的xml实现,这里我们讲讲使用注解如何实现aop呢。前面已经讲过aop的简单理解了,这里就不在赘述了。

注解方式实现aop我们主要分为如下几个步骤(自己整理的,有更好的方法的话,欢迎交流[email protected]):

  1.在切面类(为切点服务的类)前用@Aspect注释修饰,声明为一个切面类。

  2.用@Pointcut注释声明一个切点,目的是为了告诉切面,谁是它的服务对象。(此注释修饰的方法的方法体为空,不需要写功能比如 public void say(){};就可以了,方法名可以被候命的具体服务功能所以引用,它可以被理解为切点对象的一个代理对象方法)

  3.在对应的方法前用对应的通知类型注释修饰,将对应的方法声明称一个切面功能,为了切点而服务

  4.在spring配置文件中开启aop注释自动代理。如:<aop:aspectj-autoproxy/>

这样讲可能还是很抽象,那么,废话不多说,我们代码说话,代码如下:

骑士类:(看过上一篇内容的就知道骑士是什么东西了,嘿嘿嘿)

复制代码
 1 package com.cjh.aop2;
 2 
 3 import org.springframework.stereotype.Component;
 4 
 5 /**
 6  * @author 古斌
 7  *
 9  */
10 @Component("knight")
11 public class BraveKnight {
12     public void saying(){
13         System.out.println("我是骑士..(切点方法)");
14     }
15 }
复制代码

切面类:(注释主要在这里体现)

复制代码
 1 package com.cjh.aop2;
 2 
 3 import org.aspectj.lang.ProceedingJoinPoint;
 4 import org.aspectj.lang.annotation.After;
 5 import org.aspectj.lang.annotation.Around;
 6 import org.aspectj.lang.annotation.Aspect;
 7 import org.aspectj.lang.annotation.Before;
 8 import org.aspectj.lang.annotation.Pointcut;
 9 import org.springframework.stereotype.Component;
10 
11 /**
15  */
16 /**
17  * 注解方式声明aop
18  * 1.用@Aspect注解将类声明为切面(如果用@Component("")注解注释为一个bean对象,那么就要在spring配置文件中开启注解扫描,<context:component-scan base-package="com.cjh.aop2"/>
19  *      否则要在spring配置文件中声明一个bean对象)
20  * 2.在切面需要实现相应方法的前面加上相应的注释,也就是通知类型。
21  * 3.此处有环绕通知,环绕通知方法一定要有ProceedingJoinPoint类型的参数传入,然后执行对应的proceed()方法,环绕才能实现。
22  */
23 @Component("annotationTest")
24 @Aspect
25 public class AnnotationTest {
26     //定义切点
27     @Pointcut("execution(* *.saying(..))")
28     public void sayings(){}
29     /**
30      * 前置通知(注解中的sayings()方法,其实就是上面定义pointcut切点注解所修饰的方法名,那只是个代理对象,不需要写具体方法,
31      * 相当于xml声明切面的id名,如下,相当于id="embark",用于供其他通知类型引用)
32      * <aop:config>
33         <aop:aspect ref="mistrel">
34             <!-- 定义切点 -->
35             <aop:pointcut expression="execution(* *.saying(..))" id="embark"/>
36             <!-- 声明前置通知 (在切点方法被执行前调用) -->
37             <aop:before method="beforSay" pointcut-ref="embark"/>
38             <!-- 声明后置通知 (在切点方法被执行后调用) -->
39             <aop:after method="afterSay" pointcut-ref="embark"/>
40         </aop:aspect>
41        </aop:config>
42      */
43     @Before("sayings()")
44     public void sayHello(){
45         System.out.println("注解类型前置通知");
46     }
47     //后置通知
48     @After("sayings()")
49     public void sayGoodbey(){
50         System.out.println("注解类型后置通知");
51     }
52     //环绕通知。注意要有ProceedingJoinPoint参数传入。
53     @Around("sayings()")
54     public void sayAround(ProceedingJoinPoint pjp) throws Throwable{
55         System.out.println("注解类型环绕通知..环绕前");
56         pjp.proceed();//执行方法
57         System.out.println("注解类型环绕通知..环绕后");
58     }
59 }
复制代码

spring配置文件:

复制代码
 1 <?xml version="1.0" encoding="UTF-8"?>
 2 <beans xmlns="http://www.springframework.org/schema/beans"
 3     xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
 4     xmlns:context="http://www.springframework.org/schema/context"
 5     xmlns:aop="http://www.springframework.org/schema/aop"
 6     xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
 7         http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-4.3.xsd
 8         http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.3.xsd">
 9     <!-- 开启注解扫描 -->
10     <context:component-scan base-package="com.cjh.aop2"/>
11     <!-- 开启aop注解方式,此步骤s不能少,这样java类中的aop注解才会生效 -->
12     <aop:aspectj-autoproxy/>
13 </beans>
复制代码

测试代码:

复制代码
 1 package com.cjh.aop2;
 2 
 3 import org.springframework.context.ApplicationContext;
 4 import org.springframework.context.support.ClassPathXmlApplicationContext;
 5 
 6 /**
 7  * 
11  */
12 public class Test {
13     public static void main(String[] args) {
14         ApplicationContext ac = new ClassPathXmlApplicationContext("com/cjh/aop2/beans.xml");
15         BraveKnight br = (BraveKnight) ac.getBean("knight");
16         br.saying();
17     }
18 }
复制代码

运行结果:

注解类型环绕通知..环绕前
注解类型前置通知
我是骑士..(切点方法)
注解类型环绕通知..环绕后
注解类型后置通知

========================分割线===================================

  因为使用了注解方式,所以配置文件少了很多内容,只需要一句<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两个包,云盘地址:链接:http://pan.baidu.com/s/1qXQurO4 密码:nc4i

工程目录如下:(打码码的类不用管,只是为了不为大家造成误导,所以划掉了)

通知注解类型如下:

猜你喜欢

转载自www.cnblogs.com/gu-bin/p/10473417.html