`
thtwin
  • 浏览: 165891 次
  • 性别: Icon_minigender_1
  • 来自: 成都
社区版块
存档分类
最新评论

Spring框架与AOP思想的研究与应用

阅读更多
引言

Aspect Oriented Programming(AOP)是近年来计算机技术中比较热门的话题之一。其发展历史从学术领域和研发机构的运用开始,目前流行的Spring应用程序框架将AOP思想融入了整个框架的设计开发与应用当中。使用Spring框架固然给我们的编程带来了好处与便利,但是同时存在着一个问题,对于初学者来说,所谓的“控制反转”,不是一个能够望文生义的好名称,“依赖注入”也是一样,也正是因为这样,不少初学者很难在短时间内理解和掌握这些名字和他们的用法,而要使用AOP的功能也需要理解AOP,也比较难。基于以上原因,我们就会想到,能否简单地将Spring框架中运用到的优秀的理念,巧妙的运用到我们需要使用的地方,而又绕过不容易上手的Spring框架,做到一举两得呢?本文就将围绕着上述提出的问题给出作者的看法和观点。

AOP思想与面向方面的编程

AOP实际是GoF四人组设计模式的一种扩展,设计模式所追求的是降低代码之间的耦合度,增加程序的灵活性和可重用性,AOP实际上就是设计模式所追求的目标的一种实现。所谓的分离关注就是将某一通用的需求功能从不相关的类之中分离出来;同时,能够使得很多类共享一个行为,一旦行为发生变化,不必修改很多类,只要修改这个行为就可以。AOP就是这种实现分散关注的编程方法,它将“关注”封装在“方面”中。

面向对象的编程(OOP)方法是在面向过程的编程方法基础上进行的改进,而面向方面编程(AOP)方法又是在面向对象编程(OOP)方法的基础上进行改进而来的一种创新的软件开发方法。AOP和OOP虽然在字面上十分相似,但是却是面向不同领域的两种设计思想。OOP(面向对象编程)针对问题领域中以及业务处理过程中存在的实体及其属性和操作进行抽象和封装,面向对象的核心概念是纵向结构的,其目的是获得更加清晰高效的逻辑单元划分;而AOP则是针对业务处理过程中的切面进行提取,例如,某一个操作在各个模块中都有涉及,这个操作就可以看成“横切”存在于系统当中。在许多情况下,这些操作都是与业务逻辑相关性不强或者不属于逻辑操作的必须部分,而面向对象的方法很难对这种情况做出处理。AOP则将这些操作与业务逻辑分离,使程序员在编写程序时可以专注于业务逻辑的处理,而利用AOP将贯穿于各个模块间的横切关注点自动耦合进来。AOP所面对的是处理过程中的某个步骤或阶段,对不同的阶段领域加以隔离,已获得逻辑过程中各部分之间低耦合性的隔离效果,其与面向方面编程在目标上有着本质的差异。AOP的核心思想就是将应用程序中的业务逻辑处理部分同对其提供支持的通用服务,即所谓的“横切关注点”进行分离,这些“横切关注点”贯穿了程序中的多个纵向模块的需求。

使用AOP机制进行开发,首先要对方面进行了解,将需求分解成一般关注点和横切关注点,即将核心模块级的关注点和系统级的横切关注点分离;然后各自独立的实现这些关注点;最后用工具将业务逻辑代码和横切关注点代码编织到一起,形成最终的程序。通过面向方面的编程可以减少编码时间和重复。

目前已经形成的Spring框架

1、Spring框架的特点

Spring框架目前如此流行,一方面的原因在于Spring提供了一套全面并且十分成熟的轻型应用程序基本框架,并且对复杂的应用开发提供了有力的支持。除此之外,从实际应用开发角度来看,Spring最大的优势在于它是从实际项目开发经验中抽取的,其提供了丰富的类库,可大大节省编码量,它是一种高效的、可高度重用的应用框架。Spring框架中目前最吸引人也是该应用框架最具特色的地方就是名为控制反转(IOC=Inverse Of Control)或者依赖注入(DI=Dependence Injection)的设计思想,这是一种相当优秀的设计思想,即“好莱坞”原则:不用你主动来找我,我会通知你。但是,仅仅凭借着这样一个单纯的设计模式并不能使得Spring如此成功,Spring最成功的地方,还是目前使用最为广泛的AOP应用,也就是Spring中基于AOP实现的业务管理机制,也正是由于这一点,使得Spring AOP成为应用框架中极其闪光的一个亮点。

2、AOP思想在Spring框架中的体现

文章前面已经讲述了AOP的概念以及什么叫做所谓的“横切”关注点,事务管理就是J2EE应用中一个横切多个对象的横切关注点的例子。

2.1 事务管理

对于J2EE应用程序而言,事务的处理一般有两种模式:依赖特定事务资源的事务处理与依赖容器的参数化事务管理。在这里我们略去对第一种处理方式的说明,直接对第二种方式,即依赖容器的参数化事务管理来阐述笔者的观点。

Spring事务管理究竟能带给我们什么?

了解Spring的人们都知道,对于传统的基于事务资源的事务处理而言,Spring并不会产生什么影响,我们照样可以成功编写并且运行这样的代码。

对于依赖容器的参数化事务管理而言,Spring则可以用来帮助实现对事务的管理而无须使用EJB。Spring本身也是一个容器,只是相对EJB容器所要付出的代价而言,Spring属于轻量级容器,它能够替代EJB,通过使用AOP来提供声明式事务管理,即可通过Spring实现基于容器的事务管理(从本质上来讲,Spring的事务管理是基于动态AOP)。Spring与EJB最大的区别在于:第一,Spring可以为任意的Java Class实现事务管理而无须转换成标准的EJB;第二,Spring事务管理并不依赖特定的事务资源从而使得系统的应用与部署更佳灵活。

2.2动态代理机制的实现

Spring框架中所提供的AOP支持,是基于动态AOP机制实现的,即通过动态Proxy模式,在目标对象的方法调用前后插入相应的处理代码。AOP代理可以是基于JDK动态代理,也可以是基于CGLIB代理。Spring默认使用的是基于Java Dynamic Proxy模式实现,这样任何的接口都能被代理。基于Spirng框架的应用程序开发,程序员会有一种自然的倾向性来实现面向接口编程而不是类,业务对象通常也是实现一个或者多个接口,这也是一种良好的编程习惯。Spring也可以基于CGLIB实现AOP代理,这样所代理的是类而不是接口。如果一个业务对象没有实现某一个接口,那么CGLIB将被使用。

#p#

我们先来分析一下Spring事务管理机制的实现原理。由于Spring内置AOP默认使用动态代理模式实现,我们就先来分析一下动态代理模式的实现方法。动态代理模式的核心就在于代码中不出现与具体应用层相关联的接口或者类引用,如上所说,这个代理类适用于任何接口的实现。下面我们来看一个例子。

public class TxHandler implements InvocationHandler {private Object originalObject;public Object bind(Object obj) { this.originalObject = obj; return Proxy.newProxyInstance(obj.getClass().getClassLoader(),obj.getClass().getInterfaces(),this);}public Object invoke(Object proxy, Method method, Object[] args)throws Throwable { Object result = null; if (!method.getName().startsWith("save")) {  UserTransaction tx = null;  try {   tx = (UserTransaction) (new InitialContext().lookup("java/tx"));   result = method.invoke(originalObject, args);   tx.commit();  } catch (Exception ex) {   if (null != tx) {    try {     tx.rollback();    } catch (Exception e) {   }  } }} else { result = method.invoke(originalObject, args);}return result;}}



下面我们来分析一下上述代码的关键所在。

首先来看一下这段代码:

return Proxy.newProxyInstance( obj.getClass().getClassLoader(),obj.getClass().getInterfaces(),this);



java.lang.reflect.Proxy.newProxyInstance方法根据传入的接口类型(obj.getClass.getInterfaces())动态构造一个代理类实例返回,这也说明了为什么动态代理实现要求其所代理的对象一定要实现一个接口。这个代理类实例在内存中是动态构造的,它实现了传入的接口列表中所包含的所有接口。

再来分析以下代码:

public Object invoke(Object proxy, Method method, Object[] args)throws Throwable { …… result = method.invoke(originalObject, args); …… return result;}



InvocationHandler.invoke方法将在被代理类的方法被调用之前触发。通过这个方法,我们可以在被代理类方法调用的前后进行一些处理,如代码中所示,InvocationHandler.invoke方法的参数中传递了当前被调用的方法(Method),以及被调用方法的参数。同时,可以通过method.invoke方法调用被代理类的原始方法实现。这样就可以在被代理类的方法调用前后写入任何想要进行的操作。

Spring的事务管理机制实现的原理,就是通过这样一个动态代理对所有需要事务管理的Bean进行加载,并根据配置在invoke方法中对当前调用的方法名进行判定,并在method.invoke方法前后为其加上合适的事务管理代码,这样就实现了Spring式的事务管理。Spring中的AOP实现更为复杂和灵活,不过基本原理是一致的。

3.AOP思想与动态代理模式的应用实例

综上我们分析了Spring框架的事务管理机制的基本现实原理。尽管Spring框架集中体现了当前流行框架中未曾关注到的一些内容,但是,Spring框架存在晦涩难懂的致命问题。以上通过对Spring框架的一些基本实现原理的研究,给我们带来了一些启示。我们如果不直接使用庞大的Spring框架,而是将融入Spring框架中的AOP思想直接应用于程序中,既绕过了Spring框架这个高门槛,又利用了Spring框架中先进的设计理念,这样便达到了一举两得的目的。

下面我们来看一段代码,我们来编写一个Dynamic Proxy based AOP实现的实例。假设现在有一个UserDao接口和以及其实现类UserDaoImp。

UserDAO.java:public interface UserDAO { public void saveUser(User user);}UserDAOImp.java:public class UserDAOImp implements UserDAO{ public void saveUser(User user) {  …… }}



我们需要在saveUser方法中添加对一个业务对象的锁,比如在saveUser前后加锁和解锁。在不影响外部逻辑和不对现有的代码做任何改动的前提下,代理模式是一个不错的选择。但是如果有多个类似的接口,面对每个接口都要实现一个类似的Proxy,实在是一个烦琐无味的苦力过程。回想一下Spring在处理这个问题上的设计理念我们不难想到,使用动态代理模式,是这个问题的一个聪明的解决方法。

public class AOPHandler implements InvocationHandler { private static Log logger = LogFactory.getLog(AOPHandler.class); private List interceptors = null; private Object originalObject; public Object bind(Object obj) {  this.originalObject = obj;  return Proxy.newProxyInstance(obj.getClass().getClassLoader(),obj.getClass().getInterfaces().this); } public Object invoke(Object proxy, Method method, Object[] args)               throws Throwable{Object result=null  if(method.getName.startsWith(“saveUser”)){   lock();   result=method.invoke(this.originalObject,args);   unlock();  }  return result; } private void lock(){  logger.info(“lock object!”); } private void unlock(){  logger.info(“unlock object!”); }}



上述代码中并没有出现与具体应用层相关联的接口以及类的引用,所以对所有的类都适用。这就解决了用静态Proxy类实现所产生的弊端。

总结与展望

以上我们讨论了Spring框架基于动态AOP机制实现以及动态代理机制的应用,围绕着AOP的实现与应用,一直有一个热门的话题,即权限管理。Spring框架目前对AOP的支持做得相当出色,但是一直有一个尴尬的问题尚未解决,那就是目前还没有一个完备的权限管理组件。仔细想想,这并不是AOP的瓶颈,而是因为权限管理的形式过于灵活和复杂多变,系统中的权限管理逻辑多种多样各不相同,我们很难做出一个统一的管理与操作。另一方面,权限管理作为一个独立的切面显得过于庞大,需要进一步切分设计,设计过程复杂,实现难度较大。所以我们还在期待着AOP思想在权限管理方面能有突破性的应用与扩展。
分享到:
评论

相关推荐

    适用spring 实现AOP思想

    Spring AOP(面向切面编程)是Spring框架的重要组成部分,它提供了一种在不修改代码的情况下,通过插入切面来增强或改变已有功能的能力。在本文中,我们将深入探讨Spring AOP的核心概念、工作原理以及如何在实际项目...

    以注解方式模拟Spring IoC AOP

    依赖注入是Spring框架的核心特性之一,它通过反转对象创建和管理的控制权,使得应用程序组件之间的耦合度降低。在Spring中,通常通过以下三种注解实现IoC: - `@Autowired`:自动装配,Spring会根据类型或名称找到...

    小马哥讲 Spring AOP 编程思想 - API 线索图.pdf

    Spring AOP 是Spring框架提供的一个功能模块,它允许开发者将横切关注点(cross-cutting concerns)从业务逻辑中解耦出来,通过在方法调用前后进行拦截来实现。这些横切关注点通常是一些与业务逻辑不直接相关的服务...

    myeclipse spring IOC和AOP 例子

    Spring框架是Java开发中的核心组件,它通过控制反转(IOC)和面向切面编程(AOP)等特性,极大地简化了企业级应用的构建。在本教程中,我们将深入探讨这两个概念以及如何在MyEclipse环境中使用它们。下面将详细阐述...

    仿springAOP框架

    在Java开发领域,Spring框架以其强大的功能和灵活性深受开发者喜爱,其中AOP(面向切面编程)是其核心特性之一。AOP允许开发者将关注点从主业务逻辑中分离出来,如日志记录、事务管理等,实现代码的模块化和可复用性...

    使用Spring配置文件实现AOP

    在Spring框架中,面向切面编程(Aspect Oriented Programming,简称AOP)是一种强大的设计模式,它允许我们定义横切关注点,如日志、事务管理、权限检查等,然后将这些关注点与核心业务逻辑解耦。这篇教程将详细讲解...

    Spring框架研究与探讨

    ### Spring框架中的AOP思想与事务管理研究 #### 引言 随着计算机技术的发展,越来越多的设计模式被应用于软件开发中,以提高代码的可维护性和可扩展性。其中,Spring框架作为Java平台上的一个轻量级开源框架,以其...

    Spring中的AOP不生效

    在Spring框架中,AOP可以通过基于XML的配置或者基于注解的方式实现。但在实际开发过程中,可能会遇到配置好的AOP切面不生效的问题,本文将针对这一现象进行深入分析,并提供相应的解决方案。 #### 一、问题定位 当...

    理解Spring AOP实现与思想 案例代码

    Spring AOP(面向切面编程)是Spring框架中的一个重要特性,它允许开发者在不修改源代码的情况下,通过插入额外的代码(称为切面)来增强应用程序的功能。这主要通过代理模式实现,使得我们可以集中处理系统中横切...

    aop示例spring 的aop思想解决项目中多次出现的同一个问题

    Spring框架的AOP(面向切面编程)正是为了解决这类问题而设计的。AOP允许我们在不修改原有业务逻辑的情况下,将这些共性功能以一种模块化的方式插入到业务代码中,从而提高了代码的可复用性和可维护性。 首先,理解...

    spring-aop-4.2.6.RELEASE.zip

    Spring Framework Master分支则意味着这个压缩包包含了Spring框架的最新源码,开发者可以深入研究其内部机制,进行二次开发或者学习Spring的设计思想。 总的来说,"spring-aop-4.2.6.RELEASE.zip"及其关联的"spring...

    Spring及AOP应用(事务与集成)

    Spring框架的目标是简化企业级应用的开发,使得简单的JavaBean能够实现以往只有EJB才能完成的任务。 #### 二、Spring框架的体系架构 Spring框架拥有一个清晰且模块化的架构,主要由以下几个核心模块组成: 1. **...

    Spring IOC和Spring AOP_spring aop_springIOC

    面向切面编程(AOP)是Spring框架的另一大特色,它允许程序员将关注点分离,比如日志记录、事务管理、安全检查等,可以编写一次,然后在整个应用中到处使用。AOP通过切面(Aspect)和通知(Advice)来实现。 1. **...

    spring2.5AOPdemo详细资料

    Spring AOP(面向切面编程)是Spring框架的重要组成部分,它提供了一种在不修改源代码的情况下对程序进行功能增强的机制。这个"spring2.5AOPdemo详细资料"很可能是针对Spring 2.5版本的一个AOP实战示例,帮助开发者...

    Spring AOP IOC源码笔记.pdf

    Spring框架是Java开发中不可或缺的一部分,它主要由两个核心组件构成:IoC(Inversion of Control,控制反转)和AOP(Aspect Oriented Programming,面向切面编程)。本笔记将深入探讨这两个概念以及它们在Spring中...

    spring2-aop.pdf

    Spring框架通过控制反转(IoC)和面向切面编程(AOP)技术,解决了传统Java应用开发中一些常见的问题,并且支持多种编程模式和应用程序结构。 AOP(面向切面编程)是Spring框架中的一个关键特性,它允许开发者将横...

    解释Spring中的Aop代理

    在Spring框架中,AOP(面向切面编程)是一种强大的工具,它允许程序员在不修改源代码的情况下,向应用程序添加跨切面的关注点,如日志、事务管理、权限检查等。AOP代理是实现这一功能的关键机制。下面将详细解释...

    spring aop的demo

    Spring AOP(面向切面编程)是Spring框架的重要组成部分,它提供了一种在不修改源代码的情况下,对程序进行功能增强或横切关注点(如日志、事务管理、性能监控等)注入的方式。在本Demo中,我们将深入探讨Spring AOP...

    多图详解Spring框架的设计理念与设计模式

    - **解释**:Spring框架的实现不会强迫应用程序依赖于Spring的特定类库,这使得Spring可以轻松与其他框架或库集成。 5. **一站式解决方案**: - **范围**:Spring不仅关注于服务器端开发,而是适用于任何Java应用...

    Spring AOP的底层实现技术

    Spring AOP(面向切面编程)是Spring框架的重要组成部分,它提供了一种模块化和声明式的方式来处理系统中的交叉关注点,如日志、事务管理、性能监控等。AOP的核心概念包括切面(Aspect)、通知(Advice)、连接点...

Global site tag (gtag.js) - Google Analytics