博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
关于SpringAOP的XML方式的配置
阅读量:4561 次
发布时间:2019-06-08

本文共 5767 字,大约阅读时间需要 19 分钟。

AOP(XML)【理解】【应用】【重点】

       1.AOP基础实例

A.导入jar包

                     核心包(4个)         日志(2个)             AOP(4个)

                     Spring进行AOP开发(1个)(3.2资源包)

spring-aop-3.2.0.RELEASE.jar

Spring整合AspectJ框架(3.2资源包)

spring-aspects-3.2.0.RELEASE.jar

AOP联盟规范(1个) (3.0.2依赖包)

com.springsource.org.aopalliance-1.0.0.jar

aspectJ支持(1个) (3.0.2依赖包)

com.springsource.org.aspectj.weaver-1.6.8.RELEASE.jar

B.制作目标对象类(配置成Bean)

              public class UserImpl { 

              //当前类的三个方法中具有共性功能System.out.println("共性功能1");  抽取出来

              public void add(){

                     //共性功能1被抽取走了,放入到了MyAdvice类中的functionOne方法里

                     System.out.println("共性功能2");

                     System.out.println("user add....");

              }

}

注意:共性功能被抽取后,不是在原始位置进行调用,而是将共性功能删除

              C.将抽取的功能制作成通知(配置成Bean)

                     public class MyAdvice {      

              //被抽取的共性功能1

              public void functionOne(){

                     System.out.println("共性功能1");

              }    

}

说明:被抽取的共性功能制作成独立的类中方法。由于要将两个对象中的内容融合,Spring在控

制其融合的过程必须控制两个对象,因此需要分别将两个类配置为Spring管理的Bean

              D.开启AOP空间的支持

                     <beans xmlns="http://www.springframework.org/schema/beans"

                xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"

                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.xsd

                     ">

              E.配置AOP的切面

                     <!-- 配置AOP -->

              <aop:config>

                     <!-- 配置切面,指定切面类的Bean:配置切入点与通知之间的关系 -->

                     <!-- ref:配置切面对应的Bean -->

                     <aop:aspect ref="myAdvice">

                                   <!-- 声明通知的类别是何种类型  前置通知-->

<!-- 配置前置通知 -->

                                   <!-- aop:before表示在原始方法执行前追加功能 -->

                                   <!-- pointcut:配置Spring监控的方法切入点 -->

                            <!-- method:配置的切面类中对应的共性功能-方法名称 -->

                            <aop:before pointcut="execution(* cn.itcast.aop.one.UserImpl.add())" method="functionOne"/>

                     </aop:aspect>

              </aop:config>

              F.制作Spring格式的客户端程序测试

                     ApplicationContext ctx = new ClassPathXmlApplicationContext(“applicationContext.xml”);

UserDao dao = (UserDao)ctx.getBean(“userDao”);

dao.add();

       2.切入点表达式

格式:execution(切入点表达式)

              execution([方法的访问控制修饰符] 方法的返回值 包名.类名/接口名.方法名(参数))

              注意:方法的访问控制修饰符可以省略

              范例:

public void cn.itcast.UserDAO.add()    公共的cn.itcat包中的UserDAO类的无参数无返回值的add方法

void cn.itcast.UserDAO.add()                     公共的cn.itcat包中的UserDAO类的无参数无返回值的add方法

              方法的返回值可以写具体的返回值,或者写*表示任意返回值

              void cn.itcast.UserDAO.add()       公共的cn.itcat包中的UserDAO类的无参数无返回值的add方法

              * cn.itcast.UserDAO.add()    公共的cn.itcat包中的UserDAO类的无参数不限定返回值的add方法

              包名,方法名

              * cn.itcast.*.dao.UserDAO.add()   cn包下的itcast包下的任意包下的dao包下的…..

              * cn.itcast..dao.UserDAO.add()    cn包下的itcast包下的任意层包下的dao包下的…..

              * *..*.*()                                    任意包下的任意类中的任意方法

              参数

              add()              无参数

              add(*)            一个参数

              add(int)          一个int型参数

              add(*,*)         两个参数

              add(*,int)              两个参数,第一个任意,第二个int

              add(..)            任意参数

              add(*,..)         至少一个参数

              特殊格式:

              * cn.itcast.UserDAO.add(..) && args(int,int)                                  错误

              * cn.itcast.UserDAO.add(..)  &amp;&amp;  args(int,int)        正确

              * cn.itcast.UserDAO.add(int,int)  &amp;&amp;  args(a,b)      不常用,正确

              * cn.itcast.UserDAO.add(int,int)  &amp;&amp;  args(b,a)      不常用,正确

       3.切入点配置方式

切入点配置时,可以设置切面间共享切入点,也可以设置切面内共享切入点,还可以设置局部切入点

    格式一:配置在通知类别后面

           <aop:before   pointcut="execution(public void *..*.*Impl.a*())"  ….

格式二:配置在某个切面中,在当前切面范围内共享

    <aop:aspect ref="myAdvice">

                  <!-- 配置同一个切面中使用的公共切入点 -->

                  <aop:pointcut expression="execution(public void *..*.*Impl.a*())" id="pt2"/>

                  <aop:before   pointcut-ref="pt2" …..

    格式三:配置在AOP总配置中,在全部范围内共享

           <aop:config>

                  <aop:pointcut expression="execution(public void *..*.*Impl.a*())" id="pt1"/>

                  <aop:aspect ref="myAdvice">

                         <aop:before pointcut-ref="pt1"……

                  </aop:aspect>

           </aop:config>

范例:

<aop:config>

                     <!-- 所有切面共享的切入点 -->

                     <aop:pointcut expression="execution(* cn.itcast..*Tar*.*())" id="pt"/>

                     <aop:aspect ref="myAdvice">

                            <!-- 在一个切面内共享的切入点:配置独立的切入点表达式对象 -->

                            <aop:pointcut expression="execution(* cn.itcast..*Tar*.*(..))" id="pt1"/>

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

                            <aop:before pointcut="execution(* *..*.*(..))" method="fn"/>

                     </aop:aspect>

                     <aop:aspect ref="myAdvice">

                            <!-- 配置独立的切入点表达式对象 -->

                            <aop:pointcut expression="execution(* cn.itcast..*Tar*.*(..))" id="pt2"/>

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

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

                     </aop:aspect>

              </aop:config>

       4.通知类别

              通知共有五种类别

before:在原始操作前运行

       after: 在原始操作后运行,无论方法是否抛出异常

       afterReturning:在原始操作后运行,只有方法正常结束才运行,抛出异常则不运行

       afterThrowing:在原始操作中如果抛出异常,运行

       around: 在原始操作前后运行,通过ProceedingJoinPoint对象调用procee()方法完成对原始操作的调用

              //环绕通知

public void around(ProceedingJoinPoint pjp) throws Throwable{

    System.out.println("around before......");

    //调用原始方法

    pjp.proceed();

    System.out.println("around after......");

}

              通知的配置格式

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

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

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

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

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

       5.通知顺序

              在切入点之前运行的通知

                     执行允许与配置顺序有关,在上面的先运行

              在切入点之后运行的通知

                     在同一个切面中

                            执行允许与配置顺序有关,在上面的先运行

                     在不同的切面中

                            执行允许与配置顺序有关,与切面的配置顺序相反

              总结:不同通知类型执行的顺序以配置顺序为准

       6.获取通知参数

              为环绕通知之外的通知方法定义形参JoinPoint,该参数必须是通知方法的第一个参数

                     获取参数:Obejct[] args = jp.getArgs();    

                     范例:

                     public void before(JoinPoint jp){      

                            Object[] objs = jp.getArgs();

                            System.out.println("before......"+objs[0]+","+objs[1]);

                     }

              为环绕通知方法定义形参ProceedingJoinPoint对象

                     获取参数:Obejct[] args = pjp.getArgs();

       7.获取通知返回值

              afterReturning 与 around可以获取方法的返回值

              A.around通知获取返回值

                     ProceedingJoinPoint对象执行调用原始操作的返回值就是原始方法的运行返回值

                     Object res = pt.proceed(args);

                     注意:如果原始方法返回值为void类型,则around方法返回值设置为Object

如果原始方法返回值为非void类型,则around方法内必须将原始方法调用的结果返回

原始方法返回值为void类型的,通知内获取的返回值统一为null

                     public Object around(ProceedingJoinPoint pjp) throws Throwable{

                            Object res = pjp.proceed(args);

                            return res;

                     }

              B.afterReturning通知获取返回值

                     在通知方法的参数中,声明一个Object类型的参数,用于保存方法的返回值

                     public void afterReturning(JoinPoint jp,Object abc){

                            System.out.println("afterReturning......"+ abc);

                     }

                     在配置文件中,为afterReturning声明保存返回值的变量名

                     <aop:after-returning  method="afterReturning" returning="abc"/>

       8.获取通知异常对象

异常对象的获取方式与返回值很相似,声明变量,在配置中声明保存异常对象的变量名

              <aop:after-throwing pointcut-ref="pt" method="afterThrowing" throwing="e"/>

              public void afterThrowing (Throwable e){

                     System.out.println("afterThrowing......."+ e);

              }

转载于:https://www.cnblogs.com/DreamDrive/p/4091558.html

你可能感兴趣的文章
ABAP 合并单元格自建函数
查看>>
2018/01/08JAVA 基础 / 接口与继承:调用父类/子类的类方法、对象方法,访问父类的类属性、对象属性的方式汇总...
查看>>
RobotFrameWork(六)控制流之For循环
查看>>
冒泡 快速 堆排序 归并排序示例
查看>>
系统性能监控界面学习之二
查看>>
算法导论 红黑树 学习 插入(三) 图文
查看>>
mySql数据库varchar类型转int类型以及查询最大(小)值的列是varchar类型
查看>>
集合之TreeMap(含JDK1.8源码分析)
查看>>
2018/12/01 一个64位操作系统的实现 第四章 导入kernel.bin(4)
查看>>
HTML
查看>>
ORACLE创建表空间,用户及授权
查看>>
热敏网口打印机无法执行切纸指令
查看>>
壁虎书3 Classification
查看>>
壁虎书6 Decision Trees
查看>>
反射整理学习<一>(转)
查看>>
python code(1)
查看>>
利用反射生成JDK动态代理
查看>>
无奈的28句 思念的28句 痛心的28句 回忆的28句
查看>>
Django-建立网页
查看>>
iptables转发备忘
查看>>