软件151 李飞瑶

一、Spring 动态代理中的基本概念 

1、关注点(concern) 
   一个关注点可以是一个特定的问题,概念、或者应用程序的兴趣点。总而言之,应用程序必须达到一个目标 
   安全验证、日志记录、事务管理都是一个关注点 
   在oo应用程序中,关注点可能已经被代码模块化了还可能散落在整个对象模型中 
2、横切关注点(crosscutting concern) 
   如何一个关注点的实现代码散落在多个类中或方法中 
3、方面(aspect) 
   一个方面是对一个横切关注点模块化,它将那些原本散落在各处的, 
   用于实现这个关注点的代码规整在一处 
4、建议(advice)通知 
   advice是point cut执行代码,是方面执行的具体实现 
5、切入点(pointcut) 
   用于指定某个建议用到何处 
6、织入(weaving) 
   将aspect(方面)运用到目标对象的过程 
7、连接点(join point) 
  程序执行过程中的一个点

通知类型: 
  try{ 
    //前置通知 
         //环绕通知 
            //调用目标对象方法 
         //环绕通知 
    //后置通知 
  }catch(){ 
    //异常通知 
  }finally{ 
    //终止通知 
  }

流程图

二.动态代理实例

项目结构图:

IUserServ接口代码与UserServImpl实现类代码和上述代码相同

LogHandler类代码

[java] view plain copy print ?
  1. public class LogHandler implements InvocationHandler {
  2. //目标对象
  3. private Object targetObject;
  4. /**
  5. * 创建动态代理类
  6. * @return object(代理类)
  7. */
  8. public Object createProxy(Object targetObject){
  9. this.targetObject = targetObject;
  10. return Proxy.newProxyInstance(
  11. targetObject.getClass().getClassLoader(),
  12. targetObject.getClass().getInterfaces(), this);
  13. }
  14. @Override
  15. public Object invoke(Object proxy, Method method, Object[] args)
  16. throws Throwable {
  17. Object obj = null;
  18. try {
  19. beforeLog();
  20. //obj: 目标对象--->代理对象的返回值--->返回给调用者的信息
  21. //this.invoke("目标对象","代理对象给目标对象传递参数");
  22. //调用目标对象中方法
  23. obj = method.invoke(targetObject, args);
  24. afterLog();
  25. } catch (Exception e) {
  26. e.printStackTrace();
  27. }
  28. return obj;
  29. }
  30. //日志管理方法
  31. private void beforeLog(){
  32. System.out.println("开始执行");
  33. }
  34. private void afterLog(){
  35. System.out.println("执行完毕");
  36. }
  37. }

ActionTest测试类代码:

[java] view plain copy print ?
  1. public class ActionTest {
  2. public static void main(String[] args) {
  3. //创建代理对象iuserServ
  4. LogHandler handler = new LogHandler();
  5. IUserServ iuserServ = (IUserServ)handler.createProxy(new UserServImpl());
  6. iuserServ.deleteUserById(new User());
  7. }
  8. }

运行结果: 
开始执行 
******执行删除方法****** 
执行完毕 
三.Spring AOP使用(2.x版本之前)

项目结构图:

 

IUserServ接口代码与UserServImpl实现类代码和上述代码相同

配置步骤:

1、配置目标对象(applicationContext.xml)

[html] view plain copy print ?
  1. <bean id="userServTarget" class="com.tarena.biz.impl.UserServImpl"/>

2、配置通知 
(a)前置通知(BeforeLogAdvice)

[java] view plain copy print ?
  1. public class BeforeLogAdvice implements MethodBeforeAdvice {
  2. /**
  3. * Method method:调用目标对象的方法
  4. * Object[] args:发送给目标对象的参数列表
  5. * Object target:目标对象
  6. */
  7. public void before(Method method, Object[] args, Object target)
  8. throws Throwable {
  9. beforeLog();
  10. }
  11. private void beforeLog(){
  12. System.out.println("开始执行");
  13. }
  14. }

(b)后置通知(AfterLogAdvice)

[java] view plain copy print ?
  1. public class AfterLogAdvice implements AfterReturningAdvice {
  2. /**
  3. * Object returnValue:目标对象返回值
  4. *  Method method:目标对象方法名
  5. *  Object[] args:目标对象参数列表
  6. *  Object target:目标对象
  7. */
  8. public void afterReturning(Object returnValue, Method method,
  9. Object[] args, Object target) throws Throwable {
  10. afterLog();
  11. }
  12. private void afterLog(){
  13. System.out.println("执行完毕");
  14. }
  15. }

(c)在spring容器中,让容器管理通知(applicationContext.xml)

  1. <!-- 定义通知 -->
  2. <!-- 前置通知 -->
  3. <bean id="beforeLogAdvice" class="com.tarena.advice.BeforeLogAdvice"/>
  4. <!-- 后置通知 -->
  5. <bean id="afterLogAdvice" class="com.tarena.advice.AfterLogAdvice"/>

3、配置代理对象(applicationContext.xml)

  1. <!-- 代理类作用: 生成代理类,织入通知 -->
  2. <bean id="userServProxy"
  3. class="org.springframework.aop.framework.ProxyFactoryBean">
  4. <property name="interfaces">
  5. <!-- 可以添加多个接口 -->
  6. <list>
  7. <value>com.tarena.biz.IUserServ</value>
  8. </list>
  9. </property>
  10. <!-- 引入通知 -->
  11. <property name="interceptorNames">
  12. <list>
  13. <value>beforeLogAdvice</value>
  14. <value>afterLogAdvice</value>
  15. </list>
  16. </property>
  17. <!-- 目标对象 -->
  18. <property name="target" ref="userServTarget"/>
  19. </bean>

4.访问() 
Spring容器:通过代理对象调用-->织入通知--->目标对象 
程序员:访问代理对象

测试类(ActionTest):

[java] view plain copy print ?
  1. public class ActionTest {
  2. public static void main(String[] args) {
  3. ApplicationContext ac = new ClassPathXmlApplicationContext("applicationContext.xml");
  4. IUserServ iuserServ = (IUserServ)ac.getBean("userServProxy");
  5. iuserServ.deleteUserById(new User());
  6. iuserServ.findAllUser();
  7. }
  8. }

运行结果:

开始执行 
******执行删除方法****** 
执行完毕 
开始执行 
*******执行查询方法******* 
执行完毕 
四.Spring AOP使用(2.x版本之后)这种方式需要额外添加两个jar包,

存放位置在spring-framework-2.5.6.SEC01\lib\aspectj文件夹下。

项目结构图

 
IUserServ接口代码与UserServImpl实现类代码和上述代码相同

LogAdvice中

  1. public class LogAdvice {
  2. public void beforeLog(){
  3. System.out.println("开始执行");
  4. }
  5. public void afterLog(){
  6. System.out.println("执行完毕");
  7. }
  8. }

applicationContext.xml中

  1. <!-- spring2.x后 -->
  2. <!-- 目标对象 -->
  3. <bean id="userServImpl" class="com.tarena.biz.impl.UserServImpl"/>
  4. <!-- 通知 -->
  5. <bean id="logAdvice" class="com.tarena.advice.LogAdvice"/>
  6. <aop:config>
  7. <aop:aspect id="logAspect" ref="logAdvice">
  8. <!-- 切入点 -->
  9. <aop:pointcut id="beforePointCut"
  10. expression="execution(* saveUser*(..))"/>
  11. <aop:pointcut id="afterPointCut"
  12. expression="execution(* saveUser*(..))"/>
  13. <!-- 织入(通知作用于切入点) -->
  14. <aop:before method="beforeLog" pointcut-ref="beforePointCut"/>
  15. <aop:after method="afterLog" pointcut-ref="afterPointCut"/>
  16. </aop:aspect>
  17. </aop:config>

测试类:

  1. public class ActionTest {
  2. public static void main(String[] args) {
  3. ApplicationContext ac = new ClassPathXmlApplicationContext("applicationContext.xml");
  4. IUserServ iuserServ = (IUserServ)ac.getBean("userServImpl");
  5. iuserServ.deleteUserById(new User());
  6. iuserServ.findAllUser();
  7. iuserServ.saveUser(new User());
  8. }
  9. }

运行结果 
******执行删除方法****** 
*******执行查询方法******* 
开始执行 
*******执行添加方法******** 
执行完毕

注:如果要在业务层所有的方法前后添加日志文件,则需要更改为以下配置

  1. <aop:pointcut id="beforePointCut"
  2. expression="execution(* com.tarena.biz.*.*(..))"/>
  3. <aop:pointcut id="afterPointCut"
  4. expression="execution(* com.tarena.biz.*.*(..))"/>

运行结果:

开始执行 
******执行删除方法****** 
执行完毕 
开始执行 
*******执行查询方法******* 
执行完毕 
开始执行 
*******执行添加方法******** 
执行完毕

Spring AOP之动态代理的更多相关文章

  1. Spring AOP 和 动态代理技术

    AOP 是什么东西 首先来说 AOP 并不是 Spring 框架的核心技术之一,AOP 全称 Aspect Orient Programming,即面向切面的编程.其要解决的问题就是在不改变源代码的情 ...

  2. Spring AOP --JDK动态代理方式

    我们知道Spring是通过JDK或者CGLib实现动态代理的,今天我们讨论一下JDK实现动态代理的原理. 一.简述 Spring在解析Bean的定义之后会将Bean的定义生成一个BeanDefinit ...

  3. Spring AOP JDK动态代理与CGLib动态代理区别

    静态代理与动态代理 静态代理 代理模式 (1)代理模式是常用设计模式的一种,我们在软件设计时常用的代理一般是指静态代理,也就是在代码中显式指定的代理. (2)静态代理由 业务实现类.业务代理类 两部分 ...

  4. Spring AOP 前奏--动态代理

  5. 深入理解Spring AOP之二代理对象生成

    深入理解Spring AOP之二代理对象生成 spring代理对象 上一篇博客中讲到了Spring的一些基本概念和初步讲了实现方法,当中提到了动态代理技术,包含JDK动态代理技术和Cglib动态代理 ...

  6. AOP jdk动态代理

    一: jdk动态代理是Spring AOP默认的代理方法.要求 被代理类要实现接口,只有接口里的方法才能被代理,主要步骤是先创建接口,接口里创建要被代理的方法,然后定义一个实现类实现该接口,接着将被代 ...

  7. 浅谈Spring中JDK动态代理与CGLIB动态代理

    前言Spring是Java程序员基本不可能绕开的一个框架,它的核心思想是IOC(控制反转)和AOP(面向切面编程).在Spring中这两个核心思想都是基于设计模式实现的,IOC思想的实现基于工厂模式, ...

  8. AOP与动态代理有什么联系

    曾遇到“AOP与动态代理有什么联系”的问题,现把个人观点整理如下: 我觉得,动态代理是AOP的主要实现手段之一,AOP是动态代理的一种应用深化 AOP是一种思想,或者是方法论,类似OOP,是OOP的有 ...

  9. 【Java EE 学习 51】【Spring学习第三天】【cglib动态代理】【AOP和动态代理】【切入点表达式】

    一.cglib动态代理 1.简介 (1)CGlib是一个强大的,高性能,高质量的Code生成类库.它可以在运行期扩展Java类与实现Java接口. (2) 用CGlib生成代理类是目标类的子类. (3 ...

随机推荐

  1. 【codeforces 797E】Array Queries

    [题目链接]:http://codeforces.com/problemset/problem/797/E [题意] 给你一个n个元素的数组; 每个元素都在1..n之间; 然后给你q个询问; 每个询问 ...

  2. 学习笔记——TCP“三次握手”和“四次挥手”,简单介绍

    TCP/IP协议中,TCP协议提供可靠的连接服务,采用“三次握手”建立一个连接. (1)第一次握手:客户端发送SYN包(SYN=j)到服务器,并进入SYN_SEND状态,等待服务器确认. (2)第二次 ...

  3. JSP中page、request、session、application作用域的使用

    几乎所有的Web开发语言都支持Session功能,Servlet也不例外. Servlet/JSP中的Session功能是通过作用域(scope)这个概念来实现的. 作用域分为四种,分别为: page ...

  4. oracle 12c之前用sequence 和 trigger来生成自动增长的列

    SQL> create table scott.t1 (id number, val varchar2(8)); Table created. SQL> CREATE SEQUENCE s ...

  5. ZOJ 3213

    /* ZOJ 3213 好吧,看过那种括号表示法后,就崩溃了,实在受不了.情况复杂,写了两天,人也有点傻X了,只能放弃,转而用最小表示法. 最小表示法不难写: 1)首先,要承认路径上有格子不选的情况, ...

  6. Linux/CentOS各种服务框架的搭建完整流程

    在2012年的时候,由于要照应新人对Linux以及相关服务的了解和学习,我特地把当时我们创业项目的全部服务搭建过程写成了一篇文档,能够让他们学习而且有所參照. 以下就以这篇文档为底稿,进行一些改动和敏 ...

  7. UIScrollView滚动时隐藏底部导航栏问题

    - (void)scrollViewDidScroll:(UIScrollView *)scrollView { NSLog(@"開始滚动"); int currentPostio ...

  8. ZOJ 3814 Sawtooth Puzzle (2014年牡丹江赛区网络赛F题)

    1.题目描写叙述:点击打开链接 2.解题思路:本题是一道隐式图的搜索题目.一般来说,这类题目首先要定义状态,接下来是弄清楚状态怎样转移,以及状态怎样判重,怎样推断当前状态是否和目标状态同样.至于求解最 ...

  9. springboot配置文件加载位置

    springboot 启动会扫描以下位置的application.properties或者application.yml文件作为Spring boot的默认配置文件 –file:./config/ – ...

  10. Codeforces Round #330 (Div. 2) D. Max and Bike 二分

    D. Max and Bike For months Maxim has been coming to work on his favorite bicycle. And quite recently ...