spring之AOP基本概念和配置详解

整理文档很辛苦,赏杯茶钱您下走!

免费阅读已结束,点击下载阅读编辑剩下 ...

阅读已结束,您可以下载文档离线阅读编辑

资源描述

首先我们来看一下官方文档所给我们的关于AOP的一些概念性词语的解释:切面(Aspect):一个关注点的模块化,这个关注点可能会横切多个对象。事务管理是J2EE应用中一个关于横切关注点的很好的例子。在SpringAOP中,切面可以使用基于模式)或者基于Aspect注解方式来实现。通俗点说就是我们加入的切面类(比如log类),可以这么理解。连接点(Joinpoint):在程序执行过程中某个特定的点,比如某方法调用的时候或者处理异常的时候。在SpringAOP中,一个连接点总是表示一个方法的执行。通俗的说就是加入切点的那个点通知(Advice):在切面的某个特定的连接点上执行的动作。其中包括了“around”、“before”和“after”等不同类型的通知(通知的类型将在后面部分进行讨论)。许多AOP框架(包括Spring)都是以拦截器做通知模型,并维护一个以连接点为中心的拦截器链。切入点(Pointcut):匹配连接点的断言。通知和一个切入点表达式关联,并在满足这个切入点的连接点上运行(例如,当执行某个特定名称的方法时)。切入点表达式如何和连接点匹配是AOP的核心:Spring缺省使用AspectJ切入点语法。引入(Introduction):用来给一个类型声明额外的方法或属性(也被称为连接类型声明(inter-typedeclaration))。Spring允许引入新的接口(以及一个对应的实现)到任何被代理的对象。例如,你可以使用引入来使一个bean实现IsModified接口,以便简化缓存机制。目标对象(TargetObject):被一个或者多个切面所通知的对象。也被称做被通知(advised)对象。既然SpringAOP是通过运行时代理实现的,这个对象永远是一个被代理(proxied)对象。AOP代理(AOPProxy):AOP框架创建的对象,用来实现切面契约(例如通知方法执行等等)。在Spring中,AOP代理可以是JDK动态代理或者CGLIB代理。织入(Weaving):把切面连接到其它的应用程序类型或者对象上,并创建一个被通知的对象。这些可以在编译时(例如使用AspectJ编译器),类加载时和运行时完成。Spring和其他纯JavaAOP框架一样,在运行时完成织入。通知类型:前置通知(Beforeadvice):在某连接点之前执行的通知,但这个通知不能阻止连接点之前的执行流程(除非它抛出一个异常)。后置通知(Afterreturningadvice):在某连接点正常完成后执行的通知:例如,一个方法没有抛出任何异常,正常返回。异常通知(Afterthrowingadvice):在方法抛出异常退出时执行的通知。最终通知(After(finally)advice):当某连接点退出的时候执行的通知(不论是正常返回还是异常退出)。环绕通知(AroundAdvice):包围一个连接点的通知,如方法调用。这是最强大的一种通知类型。环绕通知可以在方法调用前后完成自定义的行为。它也会选择是否继续执行连接点或直接返回它自己的返回值或抛出异常来结束执行。环绕通知是最常用的通知类型。和AspectJ一样,Spring提供所有类型的通知,我们推荐你使用尽可能简单的通知类型来实现需要的功能。例如,如果你只是需要一个方法的返回值来更新缓存,最好使用后置通知而不是环绕通知,尽管环绕通知也能完成同样的事情。用最合适的通知类型可以使得编程模型变得简单,并且能够避免很多潜在的错误。比如,你不需要在JoinPoint上调用用于环绕通知的proceed()方法,就不会有调用的问题。springAOP的实现在spring2.5中,常用的AOP实现方式有两种。第一种是基于xml配置文件方式的实现,第二种是基于注解方式的实现。接下来,以具体的示例来讲解这两种方式的使用。下面我们要用到的实例是一个注册,就有用户名和密码,我们利用AOP来实现在用户注册的时候实现在保存数据之前和之后或者是抛出异常时,在这些情况下都给他加上日志。在这里我们只讲解AOP,所以我只把关键代码贴出来,不相干的就不贴了。首先我们来看一下业务逻辑service层:[java]viewplaincopyprint?1./**2.*RegisterService的实现类3.*@author曹胜欢*/4.publicclassRegisterServiceImplimplementsRegisterService{5.privateRegisterDaoregisterDao;6.publicRegisterServiceImpl(){}7./**带参数的构造方法*/8.publicRegisterServiceImpl(RegisterDaoregisterDao){9.this.registerDao=registerDao;10.}11.publicvoidsave(Stringloginname,Stringpassword){12.registerDao.save(loginname,password);13.thrownewRuntimeException(故意抛出一个异常。。。。);14.}15./**set方法*/16.publicvoidsetRegisterDao(RegisterDaoregisterDao){17.this.registerDao=registerDao;18.}}对于业务系统来说,RegisterServiceImpl类就是目标实现类,它的业务方法,如save()方法的前后或代码会出现异常的地方都是AOP的连接点。下面是日志服务类的代码:[java]viewplaincopyprint?1./**2.*日志切面类3.*@author曹胜欢4.*/5.publicclassLogAspect{6.//任何通知方法都可以将第一个参数定义为org.aspectj.lang.JoinPoint类型7.publicvoidbefore(JoinPointcall){8.//获取目标对象对应的类名9.StringclassName=call.getTarget().getClass().getName();10.//获取目标对象上正在执行的方法名11.StringmethodName=call.getSignature().getName();12.System.out.println(前置通知:+className+类的+methodName+方法开始了);13.}14.publicvoidafterReturn(){15.System.out.println(后置通知:方法正常结束了);16.}17.publicvoidafter(){18.System.out.println(最终通知:不管方法有没有正常执行完成,一定会返回的);19.}20.publicvoidafterThrowing(){21.System.out.println(异常抛出后通知:方法执行时出异常了);22.}23.//用来做环绕通知的方法可以第一个参数定义为org.aspectj.lang.ProceedingJoinPoint类型24.publicObjectdoAround(ProceedingJoinPointcall)throwsThrowable{25.Objectresult=null;26.this.before(call);//相当于前置通知27.try{28.result=call.proceed();29.this.afterReturn();//相当于后置通知30.}catch(Throwablee){31.this.afterThrowing();//相当于异常抛出后通知32.throwe;33.}finally{34.this.after();//相当于最终通知35.}36.returnresult;37.}38.}这个类属于业务服务类,如果用AOP的术语来说,它就是一个切面类,它定义了许多通知。Before()、afterReturn()、after()和afterThrowing()这些方法都是通知。下面我们就来看具体配置,首先来看一下:1.基于xml配置文件的AOP实现:这种方式在实现AOP时,有4个步骤。[html]viewplaincopyprint?1.?xmlversion=1.0encoding=UTF-8?2.beansxmlns=:xsi=:aop=:schemaLocation=6.://://://=registerDaoImplclass=com.zxf.dao.RegisterDaoImpl/9.beanid=registerServiceclass=com.zxf.service.RegisterServiceImpl10.propertyname=registerDaoImplref=RegisterDaoImpl/11./bean12.!--日志切面类--13.beanid=logAspectBeanclass=com.zxf.aspect.LogAspect/14.!--第1步:AOP的配置--15.aop:config16.!--第2步:配置一个切面--17.aop:aspectid=logAspectref=logAspectBean18.!--第3步:定义切入点,指定切入点表达式--19.aop:pointcutid=allMethod20.expression=execution(*com.zxf.service.*.*(..))/21.!--第4步:应用前置通知--22.aop:beforemethod=beforepointcut-ref=allMethod/23.!--第4步:应用后置通知--24.aop:after-returningmethod=afterReturnpointcut-ref=allMethod/25.!--第4步:应用最终通知--26.aop:aftermethod=afterpointcut-ref=allMethod/27.!--第4步:应用抛出异常后通知--28.aop:after-throwingmethod=afterThrowingpointcut-ref=allMethod/29.!--第4步:应用环绕通知--30.!--31.aop:aroundmethod=doAroundpointcut-ref=allMethod/32.--33./aop:aspect34./aop:config35./beans上述配置针对切入点应用了前置、后置、最终,以及抛出异常后通知。这样在测试执行RegisterServiceImpl类的save()方法时,控制台会有如下结果输出:前置通知:com.zxf.service.RegisterServiceImpl类的save方法开始了。针对MySQL的RegisterDao实现中的save()方法。后置通知:方法正常结束了。最终通

1 / 9
下载文档,编辑使用

©2015-2020 m.777doc.com 三七文档.

备案号:鲁ICP备2024069028号-1 客服联系 QQ:2149211541

×
保存成功