spring 之自动代理

来源:互联网 发布:java的日志框架 编辑:程序博客网 时间:2024/04/30 13:45

借助ProxyFactoryBean创建织入切面的代理子类,虽然对目标类进行了增强,但是增加了很多额外的配置。

spring提供自动代理机制,可以帮我们从烦琐的工作中解救出来。其实现机制借助于BeanPostProcessor的自动代理创建器的实现类。


自动代理创建器的继承关系:

 

代理器有三类:

1. 基于Bean的名字的自动代理创建器,例如BeanNameAutoProxyCreator.java

2. 基于Advisor(切面)匹配机制的自动代理创建器。对spring容器中的所有的Advisor扫描,并将其应用到匹配的Bean中。例如DefaultAdvisorAutoProxyCreator.java

3. 基于Bean中的AspjectJ注解标签的自动代理创建器,例如AnnotationAwareAspectJAutoProxyCreator.java


所有的自动代理创建器,都是实现了BeanPostProcessor。spring容器在实例化Bean时,BeanPostProcessor会对其加工,对满足匹配规则的Bean自动创建代理对象。


BeanNameAutoProxyCreator代码示例:

xml配置文件

[html] view plaincopyprint?
  1. <?xml version="1.0" encoding="UTF-8" ?>  
  2. <beans xmlns="http://www.springframework.org/schema/beans"  
  3.     xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:p="http://www.springframework.org/schema/p"  
  4.     xmlns:util="http://www.springframework.org/schema/util"  
  5.     xsi:schemaLocation="http://www.springframework.org/schema/beans   
  6.     http://www.springframework.org/schema/beans/spring-beans-3.0.xsd  
  7.     http://www.springframework.org/schema/util   
  8.     http://www.springframework.org/schema/util/spring-util-3.0.xsd">  
  9.   
  10.     <!-- 普通方法名匹配切面 -->  
  11.     <bean id="waiter" class="tome.sample.Advisor.Waiter" />  
  12.     <bean id="seller" class="tome.sample.Advisor.Seller" />  
  13.     <bean id="greetingAdvice" class="tome.sample.Advisor.GreetingBeforeAdvice" />  
  14.   
  15.     <!-- 通过Bean名称自动创建代理 -->  
  16.   
  17.     <bean  
  18.         class="org.springframework.aop.framework.autoproxy.BeanNameAutoProxyCreator"  
  19.         p:interceptorNames="greetingAdvice" p:optimize="true">  
  20.         <property name="beanNames" value="*er" />  
  21.   
  22.     </bean>  
  23.   
  24. </beans>  
注意:

beanNames可以使用 * 作通配符,也可以使用具体的bean名字,如

[html] view plaincopyprint?
  1. <property name="beanNames" value="waiter,seller" />  

p:interceptorNames指定一个或多个增强Bean的名称,可以是增强bean,也可以是切面bean

p:optimize 强制使用CGLib动态代理



DefaultAdvisorAutoProxyCreator能够扫描容器中的Advisor,并将Advisor自动织入到匹配的目标Bean中,即为匹配的目标Bean自动创建代理

代码实例:

xml配置文件:

[html] view plaincopyprint?
  1. <?xml version="1.0" encoding="UTF-8" ?>  
  2. <beans xmlns="http://www.springframework.org/schema/beans"  
  3.     xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:p="http://www.springframework.org/schema/p"  
  4.     xmlns:util="http://www.springframework.org/schema/util"  
  5.     xsi:schemaLocation="http://www.springframework.org/schema/beans   
  6.     http://www.springframework.org/schema/beans/spring-beans-3.0.xsd  
  7.     http://www.springframework.org/schema/util   
  8.     http://www.springframework.org/schema/util/spring-util-3.0.xsd">  
  9.   
  10.     <!-- 普通方法名匹配切面 -->  
  11.     <bean id="waiter" class="tome.sample.Advisor.Waiter" />  
  12.     <bean id="seller" class="tome.sample.Advisor.Seller" />  
  13.     <bean id="greetingAdvice" class="tome.sample.Advisor.GreetingBeforeAdvice" />  
  14.   
  15.     <bean id="greetingAdvisor" class="tome.sample.Advisor.GreetingAdvisor">  
  16.         <property name="advice" ref="greetingAdvice" />  
  17.     </bean>  
  18.   
  19.     <!--通过Advisor自动创建代理-->  
  20.     <bean  
  21.         class="org.springframework.aop.framework.autoproxy.DefaultAdvisorAutoProxyCreator" />  
  22.     </beans>  
  23.   
  24. </beans>  

测试类:

[html] view plaincopyprint?
  1. public static void main(String[] args) {  
  2.        String configPath = "tomge/sample/Advisor/beans.xml";  
  3.        ApplicationContext ctx = new ClassPathXmlApplicationContext(configPath);  
  4.        Waiter waiter = (Waiter) ctx.getBean("waiter");  
  5.        Seller seller = (Seller) ctx.getBean("seller");  
  6.        waiter.greetTo("Tom1");  
  7.        waiter.serveTo("Tom2");  
  8.        seller.greetTo("Tom3");  
  9.    }  


结果:

-----------begin---------------
GreetingBeforeAdvice is successfully called! tome.sample.Advisor.Waiter.greetTo
How are you!Tom1!
-----------end---------------
waiter greet to Tom1!
waiter serving Tom2!
seller greet to Tom3!


总结:

1. Spring采用JDK动态代理和CGLib动态代理在运行期织入增强,所以不需要要装备特殊的编译器或类装载器就可以使用AOP的功能。JDK动态代理面向的是接口层面,有很大的局限性;而CGLib不会对目标类做任何限制。JDK动态代理在创建对象时性能较好,但运行时性能较差,CGLib正好相反,所以如果创建的是singleton对象,我们比较青睐使用CGLib的动态代理

2. Spring有方法层面上有4类增强:前置增强、后置增强、环绕增强、抛出异常时的增强。引介增强是类级别的,它为目标类织入新的接口实现。

增强是一种简单的切面,作用于所有方法。

3. 切点是通过目标类名和方法名来定位一个连接点

4. 切面是切点和增强的联合体,通过ProxyFactoryBean将切面织入到不同的目标类中。当然如果要创建的代理子类太多,可以借助自动代理创建器,将容器中的Advisor自动织入到目标Bean中

0 0
原创粉丝点击