框架學習之Spring的AOP技術

一.AOP概述

1.AOP爲Aspect Oriented Programming的縮寫,意爲:面向切面編程;是一種編程範式,隸屬於軟工範疇,指導開發者如何組織程序結構;AOP最早由AOP聯盟的組織提出的,制定了一套規範;通過預編譯方式運行期動態代理實現程序功能的統一維護的一種技術;利用AOP可以對業務邏輯的各個部分進行隔離,從而使得業務邏輯各部分之間的耦合度降低,提高程序的可重用性,同時提高了開發的效率

2.AOP採取橫向抽取機制,取代了傳統縱向繼承體系重複性代碼(性能監視、事務管理、安全檢查、緩存)

3.可以在不修改源代碼的前提下,對程序進行增強

二.AOP的底層實現

1. Srping框架的AOP技術底層採用的代理技術,代理的方式提供了兩種
        1. 基於JDK的動態代理
            * 必須是面向接口的,只有實現了具體接口的類才能生成代理對象
        
        2. 基於CGLIB動態代理
            * 對於沒有實現了接口的類,也可以產生代理,產生這個類的子類的方式

2. Spring的傳統AOP中根據類是否實現接口,來採用不同的代理方式
        1. 如果實現類接口,使用JDK動態代理完成AOP
        2. 如果沒有實現接口,採用CGLIB動態代理完成AOP

三.JDK的動態代理

使用Proxy類來生成代理對象的一些代碼如下:

public class MyProxyUtils {
	public static UserDao getProxy(final UserDao dao) {
	    // 使用Proxy類生成代理對象
		UserDao proxy = (UserDao) Proxy.newProxyInstance(dao.getClass().getClassLoader(),
						dao.getClass().getInterfaces(), new InvocationHandler() {
							
        // 代理對象方法一執行,invoke方法就會執行一次
	//參數1 就是那個生成的代理對象
    	//參數2  代理對象所調用的方法對象
    	//參數3  代理對象所調用的方法對象中的參數對象數組
	public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
		if("save".equals(method.getName())){
			System.out.println("記錄日誌...");
			// 開啓事務
		}
			// 提交事務
			// 讓dao類的save或者update方法正常的執行下去
			return method.invoke(dao, args); 
		}
	});
	// 返回代理對象
	return proxy;
	}
}

四.CGLIB的代理技術

1. 引入CBLIB的開發包
        * 如果想使用CGLIB的技術來生成代理對象,那麼需要引入CGLIB的開發的jar包,在Spring框架核心包中已經引入了CGLIB的開發包了。所以直接引入Spring核心開發包即可!

2. 編寫相關的代碼

public static OrderDaoImpl getProxy(){
	// 創建CGLIB核心的類
    Enhancer enhancer = new Enhancer();
    // 設置父類
	enhancer.setSuperclass(OrderDaoImpl.class);
	// 設置回調函數
	enhancer.setCallback(new MethodInterceptor() {
	@Override
	public Object intercept(Object obj, Method method, Object[] args,
    MethodProxy methodProxy) throws Throwable {
		if("save".equals(method.getName())){
			// 記錄日誌
			System.out.println("記錄日誌了...");
		}
		return methodProxy.invokeSuper(obj, args);
	}
	});
	// 生成代理對象
	OrderDaoImpl proxy = (OrderDaoImpl) enhancer.create();
	return proxy;
}

五.AOP的相關術語

  1. Joinpoint(連接點)    -- 所謂連接點是指那些被攔截到的點。在spring中,這些點指的是方法,因爲spring只支持方法類型的連接點
    2. Pointcut(切入點)    -- 所謂切入點是指我們要對哪些Joinpoint進行攔截的定義 ,也就是我們想對那個方法做增強,這個方法叫做切入點
    3. Advice(通知/增強)    -- 所謂通知是指攔截到Joinpoint之後所要做的事情就是通知.通知分爲前置通知,後置通知,異常通知,最終通知,環繞通知(切面要完成的功能)
    4. Introduction(引介)    -- 引介是一種特殊的通知在不修改類代碼的前提下, Introduction可以在運行期爲類動態地添加一些方法或Field
    5. Target(目標對象)    -- 代理的目標對象
    6. Weaving(織入)    -- 是指把增強應用到目標對象來創建新的代理對象的過程,簡單來說也就是生成代理對象的這個過程,稱之爲織入。
    7. Proxy(代理)    -- 一個類被AOP織入增強後,就產生一個結果代理類
    8. Aspect(切面)        -- 是切入點和通知的結合,切面是以後咱們自己來編寫和配置的

六.切入點的表達式

1. 再配置切入點的時候,需要定義表達式,重點的格式如下:execution(public * *(..)),具體展開如下:
        * 切入點表達式的格式如下:
            * execution([修飾符] 返回值類型 包名.類名.方法名(參數))
        
        * 修飾符可以省略不寫,不是必須要出現的。 public 可以省略不寫
        * 返回值類型是不能省略不寫的,根據你的方法來編寫返回值。可以使用 * 代替。  例如: public * com.luo.BookDaoImpl.save()
        * 包名例如:com.luo.BookDaoImpl
            * 首先com是不能省略不寫的,但是可以使用 * 代替  例如:com.*.BookDaoImpl
            * 中間的包名可以使用 * 號代替   例如:public void *.*.BookDaoImpl.save()
            * 如果想省略中間的包名可以使用 *..*  例如: public void *..*.BookDaoImpl.save()
        
        * 類名也可以使用 * 號代替,也有類似的寫法:*DaoImpl  意思是這個類以DaoImp結尾  例如:public void com.luo.*DaoImpl.save()
        * 方法也可以使用 * 號代替 
            例如:public void com.luo.*DaoImpl.*() 通配所有方法名   
            例如: public void com.luo.*DaoImpl.save*()  通配方法以save開頭
        * 參數如果是一個參數可以使用 * 號代替,如果想代表任意參數使用 ..
            例如:public void com.luo.*DaoImpl.save(*)  統配一個參數
            例如:public void com.luo.*DaoImpl.save(..)  統配任意個數的參數

七.AOP的通知類型

   1. 前置通知
        * 在目標類的方法執行之前執行。
        * 配置文件信息:<aop:after method="before" pointcut-ref="myPointcut3"/>
        * 應用:可以對方法的參數來做校驗
    
    2. 最終通知
        * 在目標類的方法執行之後執行,如果程序出現了異常,最終通知也會執行。
        * 在配置文件中編寫具體的配置:<aop:after method="after" pointcut-ref="myPointcut3"/>
        * 應用:例如像釋放資源
    
    3. 後置通知
        * 方法正常執行後的通知。        
        * 在配置文件中編寫具體的配置:<aop:after-returning method="afterReturning" pointcut-ref="myPointcut2"/>
        * 應用:可以修改方法的返回值
    
    4. 異常拋出通知
        * 在拋出異常後通知
        * 在配置文件中編寫具體的配置:<aop:after-throwing method="afterThorwing" pointcut-ref="myPointcut3"/>    
        * 應用:包裝異常的信息
    
    5. 環繞通知
        * 方法的執行前後執行。
        * 在配置文件中編寫具體的配置:<aop:around method="around" pointcut-ref="myPointcut2"/>
        * 要注意:目標的方法默認不執行,需要使用ProceedingJoinPoint對來讓目標對象的方法執行。
八.Spring框架的AOP技術(AspectJ的XML方式)

1. 步驟一:創建JavaWEB項目,引入具體的開發的jar包
        * 先引入Spring框架開發的基本開發包
            com.springsource.org.apache.commons.logging-1.1.1.jar
            com.springsource.org.apache.log4j-1.2.15.jar
            spring-beans-4.2.4.RELEASE.jar
            spring-context-4.2.4.RELEASE.jar
            spring-core-4.2.4.RELEASE.jar
            spring-expression-4.2.4.RELEASE.jar
        * 再引入Spring框架的AOP的開發包
            * spring的傳統AOP的開發的包
                * spring-aop-4.2.4.RELEASE.jar
                * com.springsource.org.aopalliance-1.0.0.jar  這個在依賴包中找
            
            * aspectJ的開發包
                * com.springsource.org.aspectj.weaver-1.6.8.RELEASE.jar 這個在依賴包中找
                * spring-aspects-4.2.4.RELEASE.jar
                 *想要做測試的話可以引入Spring測試的包
                spring-test-4.2.4.RELEASE.jar
    
    2. 步驟二:創建Spring的配置文件,引入具體的AOP的schema約束
        <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"
               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.xsd">
    
    3. 步驟三:創建包結構,編寫具體的接口和實現類
        
        * com.luo.demo2
            * CustomerDao   -- 接口
            * CustomerDaoImpl  -- 實現類
    
    4. 步驟四:將目標類配置到Spring中
        <bean id="customerDao" class="com.luo.demo2.CustomerDaoImpl"/>

5. 步驟五:定義切面類
        public class MyAspectXml {
            // 定義通知
            public void log(){
                System.out.println("記錄日誌...");
            }
        }    
    6. 步驟六:在配置文件中定義切面類
        <bean id="myAspectXml" class="com.luo.demo2.MyAspectXml"/>

7. 步驟七:在配置文件中完成aop的配置
        <aop:config>
            <!-- 引入切面類 -->
            <aop:aspect ref="myAspectXml">
                <!-- 定義通知類型:切面類的方法和切入點的表達式 -->
                <aop:before method="log" pointcut="execution(public void com.luo.demo2.CustomerDaoImpl.save())"/>
            </aop:aspect>
        </aop:config>

8. 完成測試
        @RunWith(SpringJUnit4ClassRunner.class)
        @ContextConfiguration("classpath:applicationContext.xml")
        public class Demo3 {
            @Resource(name="customerDao")  //Java的這個註解,沒有開啓註解掃描,也是可以用的
            private CustomerDao customerDao;
            @Test
            public void run1(){
                customerDao.save();
                customerDao.update();
                customerDao.delete();
            }
        }

九.Spring框架的AOP技術(註解方式)

1. 步驟一:創建JavaWEB項目,引入具體的開發的jar包
        * 先引入Spring框架開發的基本開發包
            com.springsource.org.apache.commons.logging-1.1.1.jar
            com.springsource.org.apache.log4j-1.2.15.jar
            spring-beans-4.2.4.RELEASE.jar
            spring-context-4.2.4.RELEASE.jar
            spring-core-4.2.4.RELEASE.jar
            spring-expression-4.2.4.RELEASE.jar
        * 再引入Spring框架的AOP的開發包
            * spring的傳統AOP的開發的包
                * spring-aop-4.2.4.RELEASE.jar
                * com.springsource.org.aopalliance-1.0.0.jar
            
            * aspectJ的開發包
                * com.springsource.org.aspectj.weaver-1.6.8.RELEASE.jar
                * spring-aspects-4.2.4.RELEASE.jar
    
    2. 步驟二:創建Spring的配置文件,引入具體的AOP的schema約束
        <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"
               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.xsd">
            
        </beans>

3. 步驟三:創建包結構,編寫具體的接口和實現類
        
        * com.luo.demo2
            * CustomerDao   -- 接口
            * CustomerDaoImpl  -- 實現類
    
    4. 步驟四:將目標類配置到Spring中
        <bean id="customerDao" class="com.luo.demo2.CustomerDaoImpl"/>

5. 步驟五:定義切面類
        * 添加切面和通知的註解
            * @Aspect -- 定義切面類的註解
            
            * 通知類型(註解的參數是切入點的表達式)
                * @Before                -- 前置通知
                * @AfterReturing        -- 後置通知
                * @Around                -- 環繞通知
                * @After                -- 最終通知
                * @AfterThrowing        -- 異常拋出通知
        
        * 具體的代碼如下
            @Aspect
            public class MyAspectAnno {
                @Before(value="execution(public void com.luo.demo2.CustomerDaoImpl.save())")
                public void log(){
                    System.out.println("記錄日誌...");
                }
            }

    6. 步驟六:在配置文件中定義切面類
        <bean id="myAspectAnno" class="com.luo.demo2.MyAspectAnno"/>    
    7. 步驟七:在配置文件中開啓自動代理
        <aop:aspectj-autoproxy/> 放在最前面 
    8. 完成測試
        @RunWith(SpringJUnit4ClassRunner.class)
        @ContextConfiguration("classpath:applicationContext.xml")
        public class Demo1 {
            
            @Resource(name="customerDao")
            private CustomerDao customerDao;
            
            @Test
            public void run1(){
                customerDao.save();
                customerDao.update();
            }
        }

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章