课程链接:

1    解析

1.1  使用global advisors demo

1.2  jdk代理和cglib代理的选择

1.3  如何强制使用CGLIB实现AOP?

1.4  JDK动态代理和CGLIB字节码生成的区别?

1.5  cglib的maven依赖

1.6   CGLIB代理的工作原理

1.7     简化的proxy定义

1.8  父子bean demo

1.9  为什么要使用父子bean

 

2    代码演练

2.1  通配符使用 和 ProxyFactoryBean 连接点的选择

2.2  使用父子bean(parent标签使用)

 

 

 

1    解析

1.1  使用global advisors demo

用*做通配,匹配所有拦截器加入通知链,这种情况下只执行环绕通知

<!--  执行所有sa打头的方法,都会引入切面 -->
<!-- 实现之一:NameMatchMethodPointcut,根据方法名字进行匹配 ,自带方法 -->
 <bean id="pointcutBean" class="org.springframework.aop.support.NameMatchMethodPointcut">
    <property name="mappedNames">
        <list>
            <value>sa*</value>
        </list>
    </property>
</bean>

<bean id="defaultAdvisor" class="org.springframework.aop.support.DefaultPointcutAdvisor">
    <property name="advice" ref="moocBeforeAdvice"></property>
    <property name="pointcut" ref="pointcutBean"></property>
</bean>

<!-- 从这里开始,引入多个通知   -->
<!-- 引入的业务类不是ProxyFactoryBean ,而是getObject返回的,即 target==》bizLogicImplTarget -->
<bean id="bizLogicImpl" class="org.springframework.aop.framework.ProxyFactoryBean">

<!--     具体业务类 -->
    <property name="target">
        <ref bean="bizLogicImplTarget"/>
    </property>
    
<!--     list通知不是按照顺序执行的,而是按照通知的先后顺序执行的,先执行前置通知,然后执行环绕通知,最后执行后置通知 -->
    <property name="interceptorNames">
        <list>
<!--             <value>defaultAdvisor</value> -->
<!--             <value>moocAfterReturningAdvice</value> -->
<!--             <value>moocMethodInterceptor</value> -->
<!--             <value>moocThrowsAdvice</value> -->
            <value>mooc*</value>
        </list>
    </property>
</bean>

 

1.2  jdk代理和cglib代理的选择

a  只有类,没有接口的话,使用cglib代理。(如2.2案例中没有bizLogic接口的话,会使用cglib代理)

b  有接口,使用jdk动态代理。

c  如果我们愿意,可以强制使用cglib代理。

 

1.3  如何强制使用CGLIB实现AOP?

1.3.1  在Spring3.2中

不用把CGLIB添加到classpath中,CGLIB被重新包装并包含在Spring核心的JAR(即基于CGLIB的AOP就像JDK动态代理一样“开箱即用”)

1.3.1  在Spring3.2之前的版本

 * 添加CGLIB库,SPRING_HOME/cglib/*.jar

 * 在spring配置文件中加入<aop:aspectj-autoproxy proxy-target-class="true"/>

1.3.2  在spring3.2之后的版本

没试过


1.4  JDK动态代理和CGLIB字节码生成的区别?

* JDK动态代理只能对实现了接口的类生成代理,而不能针对类

 * CGLIB是针对类实现代理,主要是对指定的类生成一个子类,覆盖其中的方法   因为是继承,所以该类或方法最好不要声明成final 

 

1.5  cglib的maven依赖

    1. <dependency>
    2. <groupId>cglib</groupId>
    3. <artifactId>cglib</artifactId>
    4. <version>2.2.2</version>
    5. </dependency>

1.6   CGLIB代理的工作原理

 CGLIB代理的工作原理是在运行时生成目标类的子类,Spring配置这个生成的子类委托方法调用到原来的目标,子类是用来实现Decorator模式,织入通知

 

1.7     简化的proxy定义

使用父子bean定义,以及内部bean定义,可能会带来更清洁和更简洁的代理定义(抽象属性标记父bean定义为抽象的这样它不能被实例化)

 

1.8  父子bean demo

<bean id="baseProxyBean" class="org.springframework.aop.framework.ProxyFactoryBean" lazy-init="true" abstract="true">
</bean>
<bean id="bizLogicImpl" parent="baseProxyBean">
<!--     具体业务类 -->
    <property name="target">
        <ref bean="bizLogicImplTarget"/>
    </property>
    <property name="proxyInterfaces">
        <value>com.imooc.aop.api.BizLogic</value>
    </property>
<!--     list通知不是按照顺序执行的,而是按照通知的先后顺序执行的,先执行前置通知,然后执行环绕通知,最后执行后置通知 -->
    <property name="interceptorNames">
        <list>
            <value>moocBeforeAdvice</value>
            <value>moocAfterReturningAdvice</value>
            <value>moocMethodInterceptor</value>
            <value>moocThrowsAdvice</value>
<!--             <value>mooc*</value> -->
        </list>
    </property>
</bean>

 

1.9  为什么要使用父子bean

用父子bean可以简化proxy定义,多个需要被代理的类的情况下使用父子 bean 方法会使代码更好复用

 

2    代码演练

2.1  通配符使用 和 ProxyFactoryBean 连接点的选择

测试类:

package com.imooc.test.aop;

import org.junit.Test;
import org.junit.runner.RunWith;
import org.junit.runners.BlockJUnit4ClassRunner;

import com.imooc.aop.api.BizLogic;
import com.imooc.test.base.UnitTestBase;

@RunWith(BlockJUnit4ClassRunner.class)
public class TestAOPAPI extends UnitTestBase{

    public TestAOPAPI() {
        super("classpath:spring-aop-api.xml");
        // TODO Auto-generated constructor stub
    }
    
    @Test
    public void testSave(){
        BizLogic bLogic = super.getbean("bizLogicImpl");
        bLogic.sove();
    }

}

 

配置文件:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans" 
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:aop="http://www.springframework.org/schema/aop" 
xmlns:context="http://www.springframework.org/schema/context"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/aop
http://www.springframework.org/schema/aop/spring-aop-4.0.xsd
http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context.xsd"
 >

 <bean id="moocBeforeAdvice" class="com.imooc.aop.api.MoocBeforeAdvice"></bean>
     
 <bean id="moocAfterReturningAdvice" class="com.imooc.aop.api.MoocAfterReturningAdvice"></bean>
     
 <bean id="moocMethodInterceptor" class="com.imooc.aop.api.MoocMethodInterceptor"></bean>
     
 <bean id="moocThrowsAdvice" class="com.imooc.aop.api.MoocThrowsAdvice"></bean>

<bean id="bizLogicImplTarget" class="com.imooc.aop.api.BizLogicImpl"></bean> <!-- 从这里开始,引入多个通知 --> <!-- 引入的业务类不是ProxyFactoryBean ,而是getObject返回的,即 target==》bizLogicImplTarget --> <bean id="bizLogicImpl" class="org.springframework.aop.framework.ProxyFactoryBean"> <!-- 具体业务类 --> <property name="target"> <ref bean="bizLogicImplTarget"/> </property> <!-- list通知不是按照顺序执行的,而是按照通知的先后顺序执行的,先执行前置通知,然后执行环绕通知,最后执行后置通知 --> <property name="interceptorNames"> <list> <!-- <value>defaultAdvisor</value> --> <!-- <value>moocAfterReturningAdvice</value> --> <!-- <value>moocMethodInterceptor</value> --> <!-- <value>moocThrowsAdvice</value> --> <value>mooc*</value> </list> </property> </bean> </beans>

 

实现类:

package com.imooc.aop.api;

public class BizLogicImpl implements BizLogic{

    @Override
    public String sove() {
        System.out.println("syso save");
        return "BizLogicImpl save";
    }
    

}

 

接口:

package com.imooc.aop.api;

public interface BizLogic {
    String sove();
}

 

环绕通知:

package com.imooc.aop.api;

import org.aopalliance.intercept.MethodInterceptor;
import org.aopalliance.intercept.MethodInvocation;

/**
 * 引入环绕通知
 * 
 * 实现  MethodInterceptor 接口
 * @author weijingli
 *
 */
public class MoocMethodInterceptor implements MethodInterceptor{

    @Override
    public Object invoke(MethodInvocation methodinvocation) throws Throwable {
        
        //实现前置通知的方法
        //得到方法    得到类
        System.out.println("MoocMethodInterceptor1:"+methodinvocation.getMethod().getName()+"    "
        +methodinvocation.getStaticPart().getClass().getName());
        
        Object obj = methodinvocation.proceed();
        
        //实现后置通知方法
        System.out.println("MoocMethodInterceptor2:"+obj);

        
        return obj;
    }

}

 

打印日志:

四月 23, 2019 6:56:02 下午 org.springframework.context.support.ClassPathXmlApplicationContext prepareRefresh
信息: Refreshing org.springframework.context.support.ClassPathXmlApplicationContext@7fe219e7: startup date [Tue Apr 23 18:56:02 CST 2019]; root of context hierarchy
四月 23, 2019 6:56:02 下午 org.springframework.beans.factory.xml.XmlBeanDefinitionReader loadBeanDefinitions
信息: Loading XML bean definitions from class path resource [spring-aop-api.xml]
MoocMethodInterceptor1:sove    java.lang.reflect.Method
syso save
MoocMethodInterceptor2:BizLogicImpl save
四月 23, 2019 6:56:04 下午 org.springframework.context.support.ClassPathXmlApplicationContext doClose
信息: Closing org.springframework.context.support.ClassPathXmlApplicationContext@7fe219e7: startup date [Tue Apr 23 18:56:02 CST 2019]; root of context hierarchy

 

2.2  使用父子bean(parent标签使用)

测试类:

package com.imooc.test.aop;

import org.junit.Test;
import org.junit.runner.RunWith;
import org.junit.runners.BlockJUnit4ClassRunner;

import com.imooc.aop.api.BizLogic;
import com.imooc.test.base.UnitTestBase;

@RunWith(BlockJUnit4ClassRunner.class)
public class TestAOPAPI extends UnitTestBase{

    public TestAOPAPI() {
        super("classpath:spring-aop-api.xml");
        // TODO Auto-generated constructor stub
    }
    
    @Test
    public void testSave(){
        BizLogic bLogic = super.getbean("bizLogicImpl");
        bLogic.sove();
    }

}

 

配置文件:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans" 
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:aop="http://www.springframework.org/schema/aop" 
xmlns:context="http://www.springframework.org/schema/context"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/aop
http://www.springframework.org/schema/aop/spring-aop-4.0.xsd
http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context.xsd"
 >

 <bean id="moocBeforeAdvice" class="com.imooc.aop.api.MoocBeforeAdvice"></bean>
     
 <bean id="moocAfterReturningAdvice" class="com.imooc.aop.api.MoocAfterReturningAdvice"></bean>
     
 <bean id="moocMethodInterceptor" class="com.imooc.aop.api.MoocMethodInterceptor"></bean>
     
 <bean id="moocThrowsAdvice" class="com.imooc.aop.api.MoocThrowsAdvice"></bean>

 <bean id="bizLogicImplTarget" class="com.imooc.aop.api.BizLogicImpl"></bean>
 
<!--  执行所有sa打头的方法,都会引入切面 -->
<!-- 实现之一:NameMatchMethodPointcut,根据方法名字进行匹配 ,自带方法 -->
 <bean id="pointcutBean" class="org.springframework.aop.support.NameMatchMethodPointcut">
    <property name="mappedNames">
        <list>
            <value>sa*</value>
        </list>
    </property>
</bean>

<bean id="defaultAdvisor" class="org.springframework.aop.support.DefaultPointcutAdvisor">
    <property name="advice" ref="moocBeforeAdvice"></property>
    <property name="pointcut" ref="pointcutBean"></property>
</bean>

<!-- 从这里开始,引入多个通知   -->
<!-- 引入的业务类不是ProxyFactoryBean ,而是getObject返回的,即 target==》bizLogicImplTarget -->
<!-- <bean id="bizLogicImpl" class="org.springframework.aop.framework.ProxyFactoryBean"> -->

<!--     具体业务类 -->
<!--     <property name="target"> -->
<!--         <ref bean="bizLogicImplTarget"/> -->
<!--     </property> -->
    
<!--     list通知不是按照顺序执行的,而是按照通知的先后顺序执行的,先执行前置通知,然后执行环绕通知,最后执行后置通知 -->
<!--     <property name="interceptorNames"> -->
<!--         <list> -->
<!--             <value>defaultAdvisor</value> -->
<!--             <value>moocAfterReturningAdvice</value> -->
<!--             <value>moocMethodInterceptor</value> -->
<!--             <value>moocThrowsAdvice</value> -->
<!--             <value>mooc*</value> -->
<!--         </list> -->
<!--     </property> -->
<!-- </bean> -->






<bean id="baseProxyBean" class="org.springframework.aop.framework.ProxyFactoryBean" lazy-init="true" abstract="true">
</bean>
<
bean id="bizLogicImpl" parent="baseProxyBean">
<!-- 具体业务类 --> <property name="target"> <ref bean="bizLogicImplTarget"/> </property> <property name="proxyInterfaces"> <value>com.imooc.aop.api.BizLogic</value> </property> <!-- list通知不是按照顺序执行的,而是按照通知的先后顺序执行的,先执行前置通知,然后执行环绕通知,最后执行后置通知 --> <property name="interceptorNames"> <list> <value>moocBeforeAdvice</value> <value>moocAfterReturningAdvice</value> <value>moocMethodInterceptor</value> <value>moocThrowsAdvice</value> <!-- <value>mooc*</value> --> </list> </property> </bean> </beans>

 

实现类:

package com.imooc.aop.api;

public class BizLogicImpl implements BizLogic{

    @Override
    public String sove() {
        System.out.println("syso save");
        return "BizLogicImpl save";
    }
    

}

 

接口:

package com.imooc.aop.api;

public interface BizLogic {
    String sove();
}

 

前置通知:

package com.imooc.aop.api;

import java.lang.reflect.Method;

import org.springframework.aop.MethodBeforeAdvice;

public class MoocBeforeAdvice implements MethodBeforeAdvice{
    
    /**
     * 实现MethodBeforeAdvice接口,可以在连接点之前执行
     * 
     * 
     * 作用是什么?
     * 将前一章中的xml配置方式改为了实现接口的实现方式 具体细节和使用参考开发文档
     * 
     */
    @Override
    public void before(Method method, Object[] aobj, Object obj)
            throws Throwable {
        System.out.println("MoocBeforeAdvice:"+method.getName()+"     "+
                obj.getClass().getName());
    }

}

 

环绕通知:

package com.imooc.aop.api;

import org.aopalliance.intercept.MethodInterceptor;
import org.aopalliance.intercept.MethodInvocation;

/**
 * 引入环绕通知
 * 
 * 实现  MethodInterceptor 接口
 * @author weijingli
 *
 */
public class MoocMethodInterceptor implements MethodInterceptor{

    @Override
    public Object invoke(MethodInvocation methodinvocation) throws Throwable {
        
        //实现前置通知的方法
        //得到方法    得到类
        System.out.println("MoocMethodInterceptor1:"+methodinvocation.getMethod().getName()+"    "
        +methodinvocation.getStaticPart().getClass().getName());
        
        Object obj = methodinvocation.proceed();
        
        //实现后置通知方法
        System.out.println("MoocMethodInterceptor2:"+obj);

        
        return obj;
    }

}

 

后置通知:

package com.imooc.aop.api;

import java.lang.reflect.Method;

import org.springframework.aop.AfterReturningAdvice;

/**
 * 实现AfterReturningAdvice接口,可以在连接点之后执行
 * @author weijingli
 *
 */
public class MoocAfterReturningAdvice implements AfterReturningAdvice{

    /**
     * obj 作为输出
     */
    @Override
    public void afterReturning(Object obj, Method method, Object[] aobj,
            Object obj1) throws Throwable {
        System.out.println("MoocAfterReturningAdvice:"+method.getName()+
                "    类的名称:"+obj1.getClass().getName()+"    "+obj);
    }

}

 

打印日志:

四月 25, 2019 6:54:59 上午 org.springframework.context.support.ClassPathXmlApplicationContext prepareRefresh
信息: Refreshing org.springframework.context.support.ClassPathXmlApplicationContext@7fe219e7: startup date [Thu Apr 25 06:54:59 CST 2019]; root of context hierarchy
四月 25, 2019 6:54:59 上午 org.springframework.beans.factory.xml.XmlBeanDefinitionReader loadBeanDefinitions
信息: Loading XML bean definitions from class path resource [spring-aop-api.xml]
MoocBeforeAdvice:sove     com.imooc.aop.api.BizLogicImpl
MoocMethodInterceptor1:sove    java.lang.reflect.Method
syso save
MoocMethodInterceptor2:BizLogicImpl save
MoocAfterReturningAdvice:sove    类的名称:com.imooc.aop.api.BizLogicImpl    BizLogicImpl save
四月 25, 2019 6:55:00 上午 org.springframework.context.support.ClassPathXmlApplicationContext doClose
信息: Closing org.springframework.context.support.ClassPathXmlApplicationContext@7fe219e7: startup date [Thu Apr 25 06:54:59 CST 2019]; root of context hierarchy

 

posted on 2019-04-23 19:06  菜鸟乙  阅读(175)  评论(0编辑  收藏  举报