https://blog.csdn.net/wenbingoon/article/details/22888619html
Join Point:(链接点) Spring AOP中,join point就是一个方法。(通俗来说就是起做用的那个方法,具体执行的方法)java
Pointcut:(切入点) 用来指定join point(通俗来说就是描述的一组符合某个条件的join point)。一般使用pointcut表达式来限定joint point,Spring默认使用AspectJ pointcut expression language。web
Advice: 在join point上特定的时刻执行的操做,Advice有几种不一样类型,下文将会讨论(通俗地来说就是起做用的内容和时间点)。正则表达式
Introduction:给对象增长方法或者属性。spring
Target object: Advice起做用的那个对象。express
AOP proxy: 为实现AOP所生成的代理。在Spring中有两种方式生成代理:JDK代理和CGLIB代理。编程
Aspect: 组合了Pointcut与Advice,在Spring中有时候也称为Advisor。某些资料说Advisor是一种特殊的Aspect,其区别是Advisor只能包含一对pointcut和advice,可是aspect能够包含多对。AOP中的aspect能够类比于OOP中的class。安全
Weaving:将Advice织入join point的这个过程。session
Before advice: 执行在join point以前的advice,可是它不能阻止joint point的执行流程,除非抛出了一个异常(exception)。app
After returning advice: 执行在join point这个方法返回以后的advice。
After throwing advice: 执行在join point抛出异常以后的advice。
After(finally) advice: 执行在join point返回以后或者抛出异常以后的advice,一般用来释放所使用的资源。
Around advice: 执行在join point这个方法执行以前与以后的advice。
Spring AOP是基于代理机制的。上文说到,Spring AOP经过JDK Proxy和CGLIB Proxy两种方法实现代理。
若是target object没有实现任何接口,那么Spring将使用CGLIB来实现代理。CGLIB是一个开源项目,它是一个强大的,高性能,高质量的Code生成类库,它能够在运行期扩展Java类与实现Java接口。
若是target object实现了一个以上的接口,那么Spring将使用JDK Proxy来实现代理,由于Spring默认使用的就是JDK Proxy,而且JDK Proxy是基于接口的。这也是Spring提倡的面向接口编程。固然,你也能够强制使用CGLIB来进行代理,可是这样可能会形成性能上的降低。
一、若是目标对象实现了接口,默认会采用JDK的动态代理机制实现AOP
二、若是目标对象实现了接口,能够强制使用CGLIB实现AOP
三、若是目标对象没有实现接口,必须使用CGLIB生成代理,spring会自动在CGLIB和JDK动态代理之间切换
4.如何强制使用CGLIB生成代理?
* 添加CGLIB库,SPRING_HOME/lib/cglib/*.jar
* 在spring的配置文件中加入:
<aop:aspectj-autoproxy proxy-target-class="true"/>
<aop:config proxy-target-class="true">
</aop:config>
请将<aop:aspectj-autoproxy>的proxy-target-class属性设置为true:(组合使用)
<aop:aspectj-autoproxy proxy-target-class="true"/>
* JDK代理只能对实现了接口的类生成代理,而不能针对类
* CGLIB是针对类实现代理的,主要对指定的类生成一个子类,并覆盖其中的方法,
由于是继承,因此不能使用final来修饰类或方法<aop:aspectj-autoproxy proxy-target-class="true"/>
在什么状况下,Spring使用CGLIB作代理?
1.在AOP(二)基础上若是将UserManagerImpl.java修改成以下,则Spring会自动使用CGLIB作动态代理;
二、若是UserManagerImpl.java类步变,仍然实现了接口UserManager,而后咱们在配置文件中,添加一个标签:<aop:aspectj-autoproxy proxy-target-class="true"/>
Spring也会使用CGLIB作代理类:
<!--<aop:aspectj-autoproxy proxy-target-class="true"/> 支持CGLIB代理 -->
<bean id="userManager" class="com.wlh.spring.UserManagerImpl" />
<bean id="securityHandler" class="com.wlh.spring.SecurityHandler" /><!-- 切面类 -->
<aop:config>
<aop:aspect id="securityAspect" ref="securityHandler"><!-- 引用切面类 -->
<!-- 表达式方法声明匹配被切入的类及其方法 -->
<aop:pointcut id="applyMethod"
expression="execution(* com.wlh.spring.*.add*(..)) || execution(* com.wlh.spring.*.del*(..))" />
<aop:before pointcut-ref="applyMethod"
method="checkSecurity" /><!-- 声明切面类的要切入方法 -->
</aop:aspect>
</aop:config>
<bean id="duke" class="springtest.Juggler">
<constructor-arg value="15" />
</bean>
<constructor-arg value="Mike" />
<property name="id" value="18" />
</bean>
<bean id="employee" factory-bean="duke" factory-method="getYourEmployee">
而AOP技术则偏偏相反,它利用一种称为“横切”的技术,剖解开封装的对象内部,并将那些影响了多个类的公共行为封装到一个可重用模块,并将其名为“Aspect”,即切面。所谓“切面”,简单地说,就是将那些与业务无关,却为业务模块所共同调用的逻辑或责任封装起来,便于减小系统的重复代码,下降模块间的耦合度,并有利于将来的可操做性和可维护性。AOP表明的是一个横向的关系,若是说“对象”是一个空心的圆柱体,其中封装的是对象的属性和行为;那么面向切面编程的方法,就仿佛一把利刃,将这些空心圆柱体剖开,以得到其内部的消息。而剖开的切面,也就是所谓的“切面”了。而后它又以巧夺天功的妙手将这些剖开的切面复原,不留痕迹。
使用“横切”技术,AOP把软件系统分为两个部分:核心关注点和横切关注点。业务处理的主要流程是核心关注点,与之关系不大的部分是横切关注点。横切关注点的一个特色是,他们常常发生在核心关注点的多处,而各处都基本类似。好比权限认证、日志、事务处理。Aop 的做用在于分离系统中的各类关注点,将核心关注点和横切关注点分离开来。正如Avanade公司的高级方案构架师Adam Magee所说,AOP的核心思想就是“将应用程序中的商业逻辑同对其提供支持的通用服务进行分离。”
实现AOP的技术,主要分为两大类:一是采用动态代理技术,利用截取消息的方式,对该消息进行装饰,以取代原有对象行为的执行;二是采用静态织入的方式,引入特定的语法建立“切面”,从而使得编译器能够在编译期间织入有关“切面”的代码。然而异曲同工,实现AOP的技术特性倒是相同的,分别为:
一、join point(链接点):是程序执行中的一个精确执行点,例如类中的一个方法。它是一个抽象的概念,在实现AOP时,并不须要去定义一个join point。
二、point cut(切入点):本质上是一个捕获链接点的结构。在AOP中,能够定义一个point cut,来捕获相关方法的调用。
三、advice(通知):是point cut的执行代码,是执行“切面”的具体逻辑。
四、aspect(切面):point cut和advice结合起来就是aspect,它相似于OOP中定义的一个类,但它表明的更可能是对象间横向的关系。
五、introduce(引入):为对象引入附加的方法或属性,从而达到修改对象结构的目的。有的AOP工具又将其称为mixin。
六、AOP代理(AOP Proxy):AOP框架建立的对象,这个对象一般能够做为目标对象的替代品,而AOP代理提供比目标对象更增强大的功能。真实的情形是,当应用调用AOP代理的方法时,AOP代理会在本身的方法中回调目标对象的方法,从而完成应用的调用。关于AOP代理的典型例子就是Spring中的事务代理Bean。一般,目标Bean的方法不是事务性的,而AOP代理包含目标Bean的所有方法,并且这 些方法通过增强变成了事务性方法。简单地说,目标对象是蓝本,AOP代理是目标对象的增强,在目标对象的基础上,增长属性和方法,提供更强大的功能。
目标对象包含一系列切入点。切入点能够触发处理链接点集合。用户能够本身定义切入点,如使用正则表达式。AOP代理包装目标对象,在切入点处加入处理。在切入点加入的处理,使得目标对象的方法功能更强。Spring 默认使用JDK动态代理实现AOP代理,主要用于代理接口。也可使用CGLIB代理。实现类的代理,而不是接口。若是业务对象没有实现接口,默认使用 CGLIB代理。但面向接口编程是良好的习惯,尽可能不要面向具体类编程。所以,业务对象一般应实现一个或多个接口。
七、目标对象(Target Object):包含一个链接点的对象,也被称为代理对象。
八、 前置通知(Before advice):在某链接点(JoinPoint)以前执行的通知,但这个通知不能阻止链接点前的执行。ApplicationContext中在<aop:aspect>里面使用<aop:before>元素进行声明。
九、后通知(After advice) :当某链接点退出的时候执行的通知(不管是正常返回仍是异常退出)。ApplicationContext中在<aop:aspect>里面使用<aop:after>元素进行声明。
十、返回后通知(After return advice) :在某链接点正常完成后执行的通知,不包括抛出异常的状况。ApplicationContext中在<aop:aspect>里面使用<after-returning>元素进行声明。
十一、环绕通知(Around advice) :包围一个链接点的通知,相似Web中Servlet规范中的Filter的doFilter方法。能够在方法的调用先后完成自定义的行为,也能够选择不执行。ApplicationContext中在<aop:aspect>里面使用<aop:around>元素进行声明。
十二、抛出异常后通知(After throwing advice) : 在方法抛出异常退出时执行的通知。 ApplicationContext中在<aop:aspect>里面使用<aop:after-throwing>元素进行声明。
Spring2.0目前只支持使用方法调用做为链接点(join point)。
Spring 定义切入点语法:excution(modifiers-pattern?ret-type-pattern declaring-type-pattern ?name-pattern(param-pattern)throws-pattern?)
除了ret-type-pattern (即返回类型模式)、name-pattern(param-pattern)(名字模式和参数模式)外,其余模式都是可选的。返回类型模式决定了方法的返回类型必须依次匹配一个链接点(即一个方法)。使用最频繁的一个返回类型模式是*,它表明了匹配任意的返回类型。若是写明了返回类型,好比String,那么只能匹配返回String类型的链接点(方法)。名字模式匹配的是方法名。你能够用*通配符表示匹配全部方法名。参数模式中,()表示匹配了不接受任何参数的方法,而(。。)表示匹配任意数量参数的方法。模式(*)表示匹配任意类型参数的方法。模式(*,String)表示匹配:第一个为任意参数类型,第二个必须为String类型的方法。
modifiers-pattern:方法的操做权限
ret-type-pattern:返回值
declaring-type-pattern:方法所在的包
name-pattern:方法名
parm-pattern:参数名
throws-pattern:异常
下面是定义切入点的例子:
。任意公共方法的执行:
excution(public * *(。。))
。任何一个以set开头的方法执行:
excution(* set*(。。))
。AccountService接口的任意方法的执行:
excution(* com.xyz.service.AccountService.*(。。))
。定义在service包的任意方法的执行:
excution(* com.xyz.service.*.*(。。))
。定义在service包或子包的任意方法的执行:
excution(* com.xyz.service..*.*(。。))
-----------------------------------------------------华丽的分隔线----------------------------------------------
在Spring配置文件里,全部的切面和通知器都要配置在<aop:config>标签里,一个applicationContext能够包含多个<aop:config>,一个<aop:config>能够包含pointcut、advisor、aspect元素(注意必须是这个顺序)。
一、声明一个切面
<aop:config>
<aop:aspect id="myAspect" ref="myBean">
。。。。。
</aop:aspect>
</aop:config>
<bean id="myBean" class="">
。。。。。
</Bean>
说明:切面用<aop:aspect>来声明,backing bean(支持bean)用ref引用。
二、声明一个切入点
<aop:config>
<aop:pointcut id="myPointcut" expression="excution(* com.service.*.*(..))"/>
</aop:config>
三、声明一个通知
Spring2.0经过<aop:advisors>元素来支持advisors概念,大多数状况下,它将和transaction advice一块儿使用,格式以下:
<aop:config>
<aop:pointcut id="myService" expression="excution(* com.xyz.service.*.*(..))"/>
<aop:advisors pointcut-ref="myService" advice-ref="tx-advice"/>
</aop:config>
<txt:advice id="tx-advice">
<tx:attributes>
<tx:method name="inser*" propagation="REQUIRED" rollback-for="Exception"/>
<tx:method name="updat*" propagation="REQUIRED" rollback-for="Exception" />
<tx:method name="delet*" propagation="REQUIRED" rollback-for="Exception" />
<tx:method name="process*" propagation="REQUIRED" rollback-for="Exception" />
<tx:method name="*" propagation="SUPPORTS" read-only="true"/>
</tx:attributes>
</txt:advice>
说明:advisors 执行切入点方法时都要执行advice-ref引用的事务处理
经过配置织入@Aspectj切面
虽然能够经过编程的方式织入切面,可是通常状况下,咱们仍是使用spring的配置自动完成建立代理织入切面的工做。
经过aop命名空间的<aop:aspectj-autoproxy />声明自动为spring容器中那些配置@aspectJ切面的bean建立代理,织入切面。固然,spring
在内部依旧采用AnnotationAwareAspectJAutoProxyCreator进行自动代理的建立工做,但具体实现的细节已经被<aop:aspectj-autoproxy />隐藏起来了
<aop:aspectj-autoproxy />有一个proxy-target-class属性,默认为false,表示使用jdk动态代理织入加强;
当配为<aop:aspectj-autoproxy poxy-target-class="true"/>时,表示使用CGLib动态代理技术织入加强;
不过即便proxy-target-class设置为false,若是目标类没有声明接口,则spring将自动使用CGLib动态代理。
Spring AOP使用JDK动态代理或者CGLIB来为目标对象建立代理。(建议优先使用JDK的动态代理)
1 若是被代理的目标对象实现了至少一个接口,则会使用JDK动态代理。全部该目标类型实现的接口都将被代理。
2 若该目标对象没有实现任何接口,则建立一个CGLIB代理。
3 若是你但愿强制使用CGLIB代理,(例如:但愿代理目标对象的全部方法,而不仅是实现自接口的方法) 那也能够。可是须要考虑如下问题:
没法通知(advise)final方法,由于他们不能被覆写。
1 |
< aop:config proxy-target-class = "true" > |
2 |
... |
3 |
</ aop:config > |
当使用@AspectJ自动代理时要强制使用CGLIB,请将<aop:aspectj-autoproxy>的proxy-target-class属性设置为true:(组合使用)
<aop:aspectj-autoproxy proxy-target-class="true"/>
spring自动扫描机制
spring2.5为咱们引入了组件自动扫描机制,它能够在classPath路径底下寻找标注了@Component、@Service、@Controller、@Repository注解的类,并把这些类归入进spring容器中管理。它的做用和在xml文件中使用bean节点配置组件是同样的。
也就是要spring自动扫描机制只会查找指定类路径下包含@Component、@Service、@Controller、@Repository这四种注解的类。
要使用自动扫描机制,咱们须要打开如下配置信息:
一、引入context命名空间 须要在xml配置文件中配置如下信息: 同上先引入context 命名空间,同时
二、在配置文件中添加context:component-scan标签
<context:component-scan base-package="*"/>
其中base-package为须要扫描的包(含子包)。
注:
一、在使用组件扫描元素时,AutowiredAnnotationBeanPostProcessor和CommonAnnotationBeanPostProcessor会隐式地被包括进来。 也就是说,连个组件都会被自动检测并织入 - 全部这一切都不须要在XML中提供任何bean配置元数据。也就是说若是使用了context:component-scan标签,就能够不须要再引入context:annotation-config标签
<context:component-scan />还容许定义过滤器将基包下的某些类归入或排除。Spring支持如下4种类型的过滤方式:
过滤器类型 表达式范例 说明
注解 org.example.SomeAnnotation 将全部使用SomeAnnotation注解的类过滤出来
类名指定 org.example.SomeClass 过滤指定的类
正则表达式 com\.kedacom\.spring\.annotation\.web\..* 经过正则表达式过滤一些类
AspectJ表达式 org.example..*Service+ 经过AspectJ表达式过滤一些类
以正则表达式为例,我列举一个应用实例:
Java代码
<context:component-scan base-package="com.casheen.spring.annotation">
<context:exclude-filter type="regex" expression="com\.casheen\.spring\.annotation\.web\..*" />
</context:component-scan>
<context:component-scan base-package="com.casheen.spring.annotation">
<context:exclude-filter type="regex" expression="com\.casheen\.spring\.annotation\.web\..*" />
</context:component-scan>
值得注意的是<context:component-scan />配置项不但启用了对类包进行扫描以实施注释驱动Bean定义的功能,同时还启用了注释驱动自动注入的功能(即还隐式地在内部注册了AutowiredAnnotationBeanPostProcessor和CommonAnnotationBeanPostProcessor),所以当使用<context:component-scan />后,就能够将<context:annotation-config />移除了。
@Controller
@Controller("Bean的名称")
定义控制层Bean,如Action
@Service
@Service("Bean的名称")
定义业务层Bean
@Repository
@Repository("Bean的名称")
定义DAO层Bean
@Component
定义Bean, 很差归类时使用.
@Autowired (Srping提供的)
默认按类型匹配,自动装配(Srping提供的),能够写在成员属性上,或写在setter方法上
@Autowired(required=true)
必定要找到匹配的Bean,不然抛异常。 默认值就是true
@Autowired
@Qualifier("bean的名字")
按名称装配Bean,与@Autowired组合使用,解决按类型匹配找到多个Bean问题。
@Resource JSR-250提供的
默认按名称装配,当找不到名称匹配的bean再按类型装配.
能够写在成员属性上,或写在setter方法上
能够经过@Resource(name="beanName") 指定被注入的bean的名称, 要是未指定name属性, 默认使用成员属性的变量名,通常不用写name属性.
@Resource(name="beanName")指定了name属性,按名称注入但没找到bean, 就不会再按类型装配了.
@Inject 是JSR-330提供的
按类型装配,功能比@Autowired少,没有使用的必要。
@Scope("prototype")
值有:singleton,prototype,session,request,session,globalSession
@PostConstruct
至关于init-method,使用在方法上,当Bean初始化时执行。
@PreDestroy
至关于destory-method,使用在方法上,当Bean销毁时执行。
@Transactional
参考:
http://log-cd.iteye.com/blog/562056 使用AspectJ LTW(Load Time Weaving)--aop整体理解 http://blog.csdn.net/hannover100/article/details/7882893 http://tc.chinawin.net/it/softwaredev/article-24a4f.html http://www.cnblogs.com/yangy608/archive/2010/11/14/1876839.html