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);

}

关于SpringAOP的XML方式的配置的更多相关文章

  1. springmvc基础篇—通过注解的方式去配置项目

    学习了通过xml方式去配置项目后,当然要掌握更简单更灵活的注解方式哟,这是官方推荐使用的方式. 一.修改配置文件,建议大家直接使用我的配置文件 <?xml version="1.0&q ...

  2. Spring中的AOP注解方式和XML方式

    应掌握内容:1. AOP的全名2. AOP的实现原理[静态代理和动态代理]3. 注解方式的配置4. 通知类型     A. 每种通知的特点和使用方式    B. 获取各种数据,方便日后操作5. 执行表 ...

  3. Spring-注入方式(基于xml方式)

    1.基于xml方式创建对象 <!--配置User类对象的创建 --> <bean id="user" class="com.at.spring5.Use ...

  4. 转-springAOP基于XML配置文件方式

    springAOP基于XML配置文件方式 时间 2014-03-28 20:11:12  CSDN博客 原文  http://blog.csdn.net/yantingmei/article/deta ...

  5. SpringMVC(十六):如何使用编程方式替代/WEB-INF/web.xml中的配置信息

    在构建springmvc+mybatis项目时,更常用的方式是采用web.xml来配置,而且一般情况下会在web.xml中使用ContextLoaderListener加载applicationCon ...

  6. springAop:Aop(Xml)配置,Aop注解配置,spring_Aop综合案例,Aop底层原理分析

    知识点梳理 课堂讲义 0)回顾Spring体系结构 Spring的两个核心:IoC和AOP 1)AOP简介 1.1)OOP开发思路 OOP规定程序开发以类为模型,一切围绕对象进行,OOP中完成某个任务 ...

  7. spring aop注解方式与xml方式配置

    注解方式 applicationContext.xml 加入下面配置 <!--Spring Aop 启用自动代理注解 --> <aop:aspectj-autoproxy proxy ...

  8. 跟着刚哥学习Spring框架--通过XML方式配置Bean(三)

    Spring配置Bean有两种形式(XML和注解) 今天我们学习通过XML方式配置Bean 1. Bean的配置方式 通过全类名(反射)的方式   √ id:标识容器中的bean.id唯一. √ cl ...

  9. spring学习笔记 星球日one - xml方式配置bean

    ide: idea lib包的导入:http://webcache.googleusercontent.com/search?q=cache:http://zyjustin9.iteye.com/bl ...

随机推荐

  1. 导入showb时候出错--2015-12-4

    [root@cache-02 ~]# /opt/coreseek/csftweb-bash: /opt/coreseek/csftweb: is a directory[root@cache-02 ~ ...

  2. jdk源码调试功能

    JDK源码重新编译——支持eclipse调试JDK源码--转载 最近在研究jdk源码,发现debug时无法查看源码里的变量值. 因为sun提供的jdk并不能查看运行中的局部变量,需要重新编译一下rt. ...

  3. Jquery UI的datepicker插件使用方法

    原文链接;http://www.ido321.com/375.html Jquery UI是一个非常丰富的Jquery插件,并且UI的各部分插件可以独自分离出来使用,这是其他很多Jquery插件没有的 ...

  4. ACM2040

    关于亲和数的详细解释如下: http://www.kepu.net.cn/gb/basic/szsx/8/8_83/8_83_1004.htm /* 亲和数 时间限制:2000/1000 MS(JAV ...

  5. MapReduce计算模型

    MapReduce计算模型 MapReduce两个重要角色:JobTracker和TaskTracker. ​ MapReduce Job 每个任务初始化一个Job,没个Job划分为两个阶段:Map和 ...

  6. Base64 报错 的解决办法 (Base-64 字符数组或字符串的长度无效。, 输入的不是有效的 Base-64 字符串,因为它包含非 Base-64 字符、两个以上的填充字符,或者填充字符间包含非法字符。)

    Base64 报错 的解决办法, 报错如下:1. FormatException: The input is not a valid Base-64 string as it contains a n ...

  7. FIREDAC数据引擎

    以前使用过BDE.ADO.DBX等数据引擎,后来发现它们都没有UNIDAC好用, 所以在很长的一段时间内中间件都使用UNIDAC作为数据引擎. 偶然的机会,使用了DELPHI XE5自带的FIREDA ...

  8. ALM11测试计划页面图解1

    在 ALM 侧栏上的测试下方,选择测试计划. 在查看菜单中,选择测试网格或测试计划树. 在 ALM 侧栏上的测试下方,选择测试计划. 右键单击测试,并选择测试详细信息. 在测试计划树中选择主题文件夹, ...

  9. VS2015开发环境配置

    1.安装VS2015 Professional(专业版),按需勾选必要项(VC.C#.WEB.GIT) Visual Basic 2015 00322-50050-03552-AA642Microso ...

  10. C#基础知识回顾-- 反射(1)

    C#基础知识回顾-- 反射(1)   反射(reflection)是一种允许用户获得类型信息的C#特性.术语“反射”源自于它的工作方式: Type对象映射它所代表的底层对象.对Type对象进行查询可以 ...