 
         
         
         
         
        版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领
文档简介
1、原理AOP(Aspect Oriented Programming),也就是面向方面编程的技术。AOP基于IoC基础,是对OOP的有益补充。AOP将应用系统分为两部分,核心业务逻辑(Core business concerns)及横向的通用逻辑,也就是所谓的方面Crosscutting enterprise concerns,例如,所有大中型应用都要涉及到的持久化管理(Persistent)、事务管理(Transaction Management)、安全管理(Security)、日志管理(Logging)和调试管理(Debugging)等。AOP正在成为软件开发的下一个光环。使用AOP,你可以
2、将处理aspect的代码注入主程序,通常主程序的主要目的并不在于处理这些aspect。AOP可以防止代码混乱。Spring framework是很有前途的AOP技术。作为一种非侵略性的、轻型的AOP framework,你无需使用预编译器或其他的元标签,便可以在Java程序中使用它。这意味着开发团队里只需一人要对付AOP framework,其他人还是像往常一样编程。AOP概念让我们从定义一些重要的AOP概念开始。 方面(Aspect):一个关注点的模块化,这个关注点实现可能另外横切多个对象。事务管理是J2EE应用中一个很好的横切关注点例子。方面用Spring的Advisor或拦截器实现。 连
3、接点(Joinpoint):程序执行过程中明确的点,如方法的调用或特定的异常被抛出。 通知(Advice):在特定的连接点,AOP框架执行的动作。各种类型的通知包括“around”、“before”和“throws”通知。通知类型将在下面讨论。许多AOP框架包括Spring都是以拦截器做通知模型,维护一个“围绕”连接点的拦截器链。 切入点(Pointcut):指定一个通知将被引发的一系列连接点的集合。AOP框架必须允许开发者指定切入点,例如,使用正则表达式。 引入(Introduction):添加方法或字段到被通知的类。Spring允许引入新的接口到任何被通知的对象。例如,你可以使用一个引入使
4、任何对象实现IsModified接口,来简化缓存。 目标对象(Target Object):包含连接点的对象,也被称作被通知或被代理对象。 AOP代理(AOP Proxy):AOP框架创建的对象,包含通知。在Spring中,AOP代理可以是JDK动态代理或CGLIB代理。 编织(Weaving):组装方面来创建一个被通知对象。这可以在编译时完成(例如使用AspectJ编译器),也可以在运行时完成。Spring和其他纯Java AOP框架一样,在运行时完成织入。各种通知类型包括: Around通知:包围一个连接点的通知,如方法调用。这是最强大的通知。Aroud通知在方法调用前后完成自
5、定义的行为,它们负责选择继续执行连接点或通过返回它们自己的返回值或抛出异常来短路执行。 Before通知:在一个连接点之前执行的通知,但这个通知不能阻止连接点前的执行(除非它抛出一个异常)。 Throws通知:在方法抛出异常时执行的通知。Spring提供强制类型的Throws通知,因此你可以书写代码捕获感兴趣的异常(和它的子类),不需要从Throwable或Exception强制类型转换。 After returning通知:在连接点正常完成后执行的通知,例如,一个方法正常返回,没有抛出异常。Around通知是最通用的通知类型。大部分基于拦截的AOP框架(如
6、Nanning和Jboss 4)只提供Around通知。如同AspectJ,Spring提供所有类型的通知,我们推荐你使用最 为合适的通知类型来实现需要的行为。例如,如果只是需要用一个方法的返回值来更新缓存,你最好实现一个after returning通知,而不是around通知,虽然around通知也能完成同样的事情。使用最合适的通知类型使编程模型变得简单,并能减少潜在错 误。例如,你不需要调用在around通知中所需使用的MethodInvocation的proceed()方法,因此就调用失败。切入点的概念是AOP的关键,它使AOP区别于其他使用拦截的技术。切入点使通知独立于OO的层次选定
7、目标。例如,提供声明式事务管理的around通知可以被应用到跨越多个对象的一组方法上。 因此切入点构成了AOP的结构要素。 拦截器(也称拦截机) 拦截机 (Interceptor), 是 AOP (Aspect-Oriented Programming) 的另一种叫法。AOP本身是一门语言,只不过我们使用的是基于JAVA的集成到Spring 中的 SpringAOP。同样,我们将通过我们的例子来理解陌生的概念。 接口类Java代码 1. <span style="font-size:
8、160;medium;">package com.test.TestSpring3; 2. 3. public interface UserService / 被拦截的接口 4. . 5. public void printUser(String user); 6. 7. </span>
9、0; 实现类Java代码 1. <span style="font-size: medium;">package com.test.TestSpring3; 2. 3. public class UserServiceImp implements UserService / 实现UserService接口 4. . 5.
10、60;public void printUser(String user) . 6. System.out.println("printUser user:" + user);/ 显示user 7. 8. 9. 10. </span&
11、gt; AOP拦截器Java代码 1. <span style="font-size: medium;">package com.test.TestSpring3; 2. 3. import ercept.MethodInterceptor; 4. import ercept.MethodInvocation;
12、160;5. 6. public class UserInterceptor implements MethodInterceptor 7. / AOP方法拦截器 8. . 9. 10. public Object invoke(MethodInvocation arg0) throws Throwable .
13、160;11. 12. try . 13. 14. if (arg0.getMethod().getName().equals("printUser") 15.
14、60; / 拦截方法是否是UserService接口的printUser方法 16. . 17. Object args
15、;= arg0.getArguments();/ 被拦截的参数 18. System.out.println("user:" + args0); 19.
16、60; arg0.getArguments()0 = "hello!"/ 修改被拦截的参数 20. 21. 22. 23. Syste
17、m.out.println(arg0.getMethod().getName() + "-!"); 24. return ceed();/ 运行UserService接口的printUser方法 25. 26.
18、160;catch (Exception e) . 27. throw e; 28. 29. 30. 31. </span><span
19、160;style="font-size: medium;"> 32. </span> 测试类Java代码 1. <span style="font-size: medium;">package com.test.TestSpring3; 2. 3. import org.springframework.beans.factory.BeanFactory;
20、0; 4. 5. import org.springframework.beans.factory.xml.XmlBeanFactory; 6. import org.springframework.context.ApplicationContext; 7. import org.springframework.context.support.ClassPathXmlApplicationContext; 8. import org.springfram
21、ework.context.support.FileSystemXmlApplicationContext; 9. import org.springframework.core.io.ClassPathResource; 10. import org.springframework.core.io.Resource; 11. import org.springframework.web.context.support.WebApplicationContextUtils; 1
22、2. 13. public class TestInterceptor . 14. 15. public static void main(String args) . 16. ApplicationContext ctx = new File
23、SystemXmlApplicationContext( 17. "classpath:applicationContext.xml"); 18. / ApplicationContext ctx = new
24、ClassPathXmlApplicationContext("applicationContext.xml"); 19. 20. UserService us = (UserService) ctx.getBean("userService&
25、quot;); 21. us.printUser("shawn"); 22. 23. 24. </span><span style="font-size: medium;"> 25. </span> 配置文件
26、160; Xml代码 1. <span style="font-size: medium;"><?xml version="1.0" encoding="UTF-8"?> 2. <!DOCTYPE beans PUBLIC "-/SPRING/DTD BEAN/EN" "/
27、dtd/spring-beans.dtd"> 3. <beans> 4. <bean id="userServiceImp" 5. class="com.test.TestSpring3.UserServiceImp" /> 6. 7.
28、60; <bean id="userInterceptor" class="com.test.TestSpring3.UserInterceptor" /> 8. 9. <bean id="userService" 10. class=
29、"org.springframework.aop.framework.ProxyFactoryBean"> 11. <!- 代理接口 -> 12. <property name="proxyInterfaces"> 13.
30、60; <value>com.test.TestSpring3.UserService</value> 14. </property> 15. <!- 目标实现类 -> 16. &
31、#160; <property name="target"> 17. <ref local="userServiceImp" /> 18. </propert
32、y> 19. <!- 拦截器 -> 20. <property name="interceptorNames"> 21.
33、0;<list> 22. <value>userInterceptor</value> 23. </list> 24.
34、60; </property> 25. </bean> 26. 27. </beans> 28. </span> 输出: user:shawn printUser-! printUser user:hello! 结论:调用方法的时
35、候 传入的值被拦截修改了.拦截器中的事务管理(事务拦截机) 如果不采用拦截机的机制时,在使用JDBC进行数据库访问时,存在两种情况: · 自动提交 这是JDBC驱动默认的模式,每次数据库操作(CRUD)成功完成后,都作为一个单独的事务自动提交,如果未成功完成,即抛出了 SQLException 的话,仅最近的一个操作将回滚。 · 非自动提交 这是想更好的控制事务时需要程序地方式进行控制: o 在进行该事务单元的任何操作之前 setAut
36、oCommit(false) o 在成功完成事务单元后 commit() o 在异常发生后 rollback()自动提交模式是不被推荐的,因为每个操作都将产生一个事务点,这对于大的应用来说性能将受到影响;再有,对于常见的业务逻辑,这种模式显得无能为力。比如:转帐,从A帐户取出100元,将其存入B帐户;如果在这两个操作之间发生了错误,那么用户A将损失了100元,而本来应该给帐户B的,却因为失败给了银行。所以,建议在所有的应用中,如果使用 JDBC 都将不得不采用非自动提交模式(你们要能发现了在我们的 JDBC 那个例子中,我们采用的就是自动提交模式,我们是为了把精力放在JDBC上,而不是事务处理
37、上),即我们不得不在每个方法中:Java代码 1. <span style="font-size: medium;">try 2. / 在获得连接后,立即通过调用setAutoCommit(false) 将事务处理置为非自动提交模式 / Prepare Query to fetch the user Information
38、60; 3. pst = conn.prepareStatement(findByName); 4. /
39、. mit(); 5. catch(Exception ex) 6. conn.rollback(); &
40、#160; 7. throw ex; 8. finally 9. try 10.
41、 / Close Result Set and Statement 11. if (rset != null) rset.close();
42、160; 12. if (pst != null) pst.close();
43、60; 13. catch (Exception ex) 14. ex.printStackTrace();
44、0; 15. throw new Exception("SQL Error while closing objects = " + ex.toString();
45、; 16. 17. 18. </span> 这样代码在AOP的倡导者看来是“肮脏”的代码。他们认为,所有的与事务有关的方法都应当可以集中配置(见声明性事务控制),并自动拦截,程序应当关心他们的主要任务,即商业逻辑,而不应和事务处理的代码搅和在一起。我先看看 Spring 是怎么做到拦截的:1 Spring 内置支持的事务处理拦截机这里因为要用到JpetStore项目中的代码,我们将 applicationCo
46、ntext.xml 全部内容列出:<?xml version="1.0" encoding="UTF-8"?><!- - Application context definition for JPetStore's business layer. - Contains bean references to the transaction manager and to the DAOs in - dataAccessContext-local/jta.xml (see web.xml's "contextConf
47、igLocation"). Jpetstore 的应用上下文定义,包含事务管理和引用了在 dataAccessContext-local/jta.xml(具体使用了哪个要看 web.xml 中的 'contextConfigLocation' 的配置)中注册的DAO-><beans xmlns="/schema/beans" xmlns:xsi="/2001/XMLSchema-instance" xmlns:aop="
48、;/schema/aop" xmlns:tx="/schema/tx" xsi:schemaLocation="/schema/beans /schema/beans/spring-beans-2.0.xsd /schema/aop http:/www.springfr
49、/schema/aop/spring-aop-2.0.xsd/schema/tx /schema/tx/spring-tx-2.0.xsd"> <!- = GENERAL DEFINITIONS = -> <!- Configurer that replaces $. placeholders with values from properties files 占位符的值将从列出的属性文件中抽取出来 -> <!-
50、 (in this case, mail and JDBC related properties) -> <bean id="propertyConfigurer" class="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer"> <property name="locations"> <list> <value>WEB-INF/perties</value>
51、<value>WEB-INF/perties</value> </list> </property> </bean> <!- MailSender used by EmailAdvice 指定用于发送邮件的 javamail 实现者,这里使用了 spring 自带的实现。此 bean 将被 emailAdvice 使用 -> <bean id="mailSender" class="org.springframework.mail.javamail.JavaMailSen
52、derImpl"> <property name="host" value="$mail.host"/> </bean> <!- = BUSINESS OBJECT DEFINITIONS = -> <!- 不需要,因为被 SpringMVC 的实现使用 Generic validator for Account objects, to be used for example by the Spring web tier -> <bean id="accountValidat
53、or" class="org.springframework.samples.jpetstore.domain.logic.AccountValidator"/> <!- 不需要,因为被 SpringMVC 的实现使用 Generic validator for Order objects, to be used for example by the Spring web tier -> <bean id="orderValidator" class="org.springframework.samples.j
54、petstore.domain.logic.OrderValidator"/> <!- 主要的商业逻辑对象,即我们所说的门面对象 注入了所有的DAO,这些DAO是引用了 dataAccessContext-xxx.xml 中定义的DAO 门面对象中的所有方法的事务控制将通过下面的 aop:config 来加以控制 - JPetStore primary business object (default implementation). - Transaction advice gets applied through the AOP configuration below
55、. -> <bean id="petStore" class="org.springframework.samples.jpetstore.domain.logic.PetStoreImpl"> <property name="accountDao" ref="accountDao"/> <property name="categoryDao" ref="categoryDao"/> <property name="p
56、roductDao" ref="productDao"/> <property name="itemDao" ref="itemDao"/> <property name="orderDao" ref="orderDao"/> </bean> <!- = ASPECT CONFIGURATION = -> <!- AOP配置,用来控制哪些方法将需要进行事务处理,采用了AspectJ 的语法 -> <aop:conf
57、ig> <!- This definition creates auto-proxy infrastructure based on the given pointcut, expressed in AspectJ pointcut language. Here: applying the advice named "txAdvice" to all methods on classes named PetStoreImpl. -> <!- 指出在 PetStoreFacade 的所有方法都将采用 txAdvice(在紧接着的元素中定义了)事务方针,
58、注意,我们这里虽然指定的是接口 PetStoreFacace, 但其暗示着其所有的实现类也将同样具有这种性质,因为本身就是实现类的方法在执行的,接口是没有方法体的。 -> <aop:advisor pointcut="execution(* *.PetStoreFacade.*(.)" advice-ref="txAdvice"/> <!- This definition creates auto-proxy infrastructure based on the given pointcut, expressed in Aspe
59、ctJ pointcut language. Here: applying the advice named "emailAdvice" to insertOrder(Order) method of PetStoreImpl -> <!- 当执行 PetStoreFacade.insertOrder方法,该方法最后一个参数为Order类型时(其实我们的例子中只有一个 insertOrder 方法,但这告诉了我们,当我们的接口或类中有重载了的方法,并且各个重载的方法可能使用不同的拦截机机制时,我们可以通过方法的参数加以指定),将执行emailAdvice(在最后
60、定义的那个元素)-> <aop:advisor pointcut="execution(* *.PetStoreFacade.insertOrder(*.Order)" advice-ref="emailAdvice"/> </aop:config> <!- 事务方针声明,用于控制采用什么样的事务策略 Transaction advice definition, based on method name patterns. Defaults to PROPAGATION_REQUIRED for all methods
61、 whose name starts with "insert" or "update", and to PROPAGATION_REQUIRED with read-only hint for all other methods. -> <tx:advice id="txAdvice"> <tx:attributes> <tx:method name="insert*"/> <tx:method name="update*"/> <t
62、x:method name="*" read-only="true"/> </tx:attributes> </tx:advice> <!- 拦截机,用于在适当的时机(通过AOP配置,如上面)在方法执行成功后发送邮件 AOP advice used to send confirmation email after order has been submitted -> <!- -> <bean id="emailAdvice" class="org.springfr
63、amework.samples.jpetstore.domain.logic.SendOrderConfirmationEmailAdvice"> <property name="mailSender" ref="mailSender"/> </bean> <!- = 忽略 REMOTE EXPORTER DEFINITIONS = -> </beans> 这个配置比想象的要简单的多:Xml代码 1. <span style="fo
64、nt-size: medium;"><aop:config> 2. <!- This definition creates auto-proxy infrastructure based on the given pointcut, expressed in AspectJ pointcut
65、60;language. 3. Here: applying the advice named "txAdvice" to all methods on classes named PetStoreImpl. 指出在 PetStoreFacade 4. 的所有方法都将采用 tx
66、Advice(在紧接着的元素中定义了)事务方针,注意,我们这里虽然指定的是接口 PetStoreFacace, 5. 但其暗示着其所有的实现类也将同样具有这种性质,因为本身就是实现类的方法在执行的,接口是没有方法体的。 -> 6. <aop:advisor
67、160;pointcut="execution(* *.PetStoreFacade.*(.)" advice-ref="txAdvice"/> 7. <!- 其它拦截机-> 8. </aop:config> 9. &
68、lt;/span> 1. 所有的拦截机配置都放在 <aop:config> 配置元素中.2. 下面还是需要理解一下几个有关AOP的专用名词,不过,是挺抽象的,最好能会意出其的用意· pointcut 切入点,比如:updateAccount 方法需要进行事务管理,则这个切入点就是“执行方法体”(execution)。Spring 所有支持的切入点类型在都在 Spring reference: . Supported Pointcut Designators 中列出了。 · advice
69、 要对这个切入点进行什么操作,比如事务控制 · advisor Spring 特有的概念,将上两个概念合到一个概念中来,即一个 advisor 包含了一个切入点及对这个切入点所实施的操作。 因为 方法执行切入点 execution 为最常见的切入点类型,我们着重介绍一下,execution 的完全形式为:execution(modifiers-pattern? ret-type-pattern declaring-type-pattern? name-pattern(param-pattern) throws-pattern?)这是一个正则表达式,其中由
70、39;?' 结尾的部分是可选的。翻译过来就是:执行(方法访问修饰符? 方法返回类型 声明类型? 方法名(方法参数类型) 抛出异常?)所有的这些都是用来定义执行切入点,即那些方法应该被侯选为切入点:· 方法访问修饰符 即 public, private 等等 · 方法返回类型 即方法返回的类型,如 void, String 等等 · 声明类型 &
71、#160; 1.5的语法,现在可以先忽略它 · 方法名 方法的名字 · 方法参数类型 方法的参数类型 · 抛出异常
72、0; 方法声明的抛出的异常 例如,所有dao代码被定义在包 com.xyz.dao及子包 com.xyz.dao.hibernate, 或者其它,如果还有的话,子包中, 里面定义的是提供DAO功能的接口或类,那么表达式:execution(* com.xyz.dao.*.*(.)表示切入点为:执行定义在包 com.xyz.dao 及其子包(因为 . 所致) 中的任何方法详细情况可以参见 Spring refernce: . Examples因此这个表达式为执行定义在类 PetStoreFacade及其实现类中的所有方法,采取的动作定义在 txA
73、dvice 中. 关于该 advice 的定义,(见声明性事务控制)一节<aop:advisor pointcut="execution(* *.PetStoreFacade.*(.)" advice-ref="txAdvice"/> 2 Spring 自定拦截机来为了进行事务控制,我们只需简单地配置几下,所有的工作都由 Spring 来做。这样固然很好,但有时我们需要有我们特有的控制逻辑。因为Spring 不可能包含所有人需要的所有拦截机。所以它提供了通过程序的方式加以定制的方式。我们的项目中就有这么一个拦截机,在用户确认付款后,
74、将定单信息通过 email 的方式发送给注册用户的邮箱中。<aop:config> . <!- 当执行 PetStoreFacade.insertOrder方法,该方法最后一个参数为Order类型时(其实我们的例子中只有一个 insertOrder 方法,但这告诉了我们,当我们的接口或类中有重载了的方法,并且各个重载的方法可能使用不同的拦截机机制时,我们可以通过方法的参数加以指定),将执行emailAdvice(在最后定义的那个元素)-> <aop:advisor pointcut="execution(* *.PetStoreFacade.insert
75、Order(*.Order)" advice-ref="emailAdvice"/> </aop:config>红色的注释已经说的很清楚这个 Advisor 了,它的切入点(pointcut) 为 PetStoreFacade 的 void insertOrder(Order order) 方法,采取的动作为引用的 emailAdvice, 下面我们就来看看 emailAdvice: <bean id="emailAdvice" class="org.springframework.samples.jpetst
76、ore.domain.logic.SendOrderConfirmationEmailAdvice"> <property name="mailSender" ref="mailSender"/> </bean>它给了这个 advice 的实现类为 logic 包中 SendOrderConfirmationEmailAdvice, 该Bean 引用了我们前面定义的邮件发送器(一个 Spring 内置的邮件发送器).下面看看这个实现类:public class SendOrderConfirmationEmailAdvice implements AfterReturningAdvice, InitializingBean / user jes on localhost private static final String DEFAULT_MAIL_FROM = "" private static final String DEFAULT_SUBJECT = "Thank you for your order!" private
温馨提示
- 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
- 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
- 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
- 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
- 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
- 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
- 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。
最新文档
- 小学语文二年级教案日记两则教学设计(2025-2026学年)
- 高一政治经济生活教案(2025-2026学年)
- 眉山中考试卷真题及答案
- 一级市政考试题目及答案
- 银行话务员考试题及答案
- 小学内初班考试题型及答案
- 深林消防员考试题及答案
- 护理记录单PDCA课件
- 莱芜市农村信用社联合社秋季校园招聘笔试备考题库(浓缩500题)及答案详解(真题汇编)
- 安顺市农村信用社联合社秋季校园招聘笔试备考题库(浓缩500题)附答案详解(完整版)
- 第一讲 决胜“十四五”奋发向前行
- 生产计划员月度汇报
- 2025年及未来5年中国特钢棒材制造市场深度评估及行业投资前景咨询报告
- 幼儿园课件:《体能大循环的有效开展策略》
- 2025见证取样员考试试题带答案
- 医疗器械委托生产课件
- 招投标自查报告和问题整改指导
- 《人工智能语言与伦理》章节测试题及答案
- 高校实验室安全基础(华东理工大学)学习通网课章节测试答案
- 建筑垃圾现场分拣作业流程方案
- 2022利达消防LD6901 消防控制室图形显示装置(Linux)使用说明书
 
            
评论
0/150
提交评论