1. 程式人生 > >Spring3實戰第一章 Aop 切面 XML配置

Spring3實戰第一章 Aop 切面 XML配置

utf-8 生成 ini 切換 動態 部分 重用 pat 轉換

剛看spring3實戰書籍第一章 切面以前沒有關註過 現在看到了 隨手試驗一下

AOP

AOP(Aspect Oriented Programming),即面向切面編程,可以說是OOP(Object Oriented Programming,面向對象編程)的補充和完善。OOP引入封裝、繼承、多態等概念來建立一種對象層次結構,用於模擬公共行為的一個集合。不過OOP允許開發者定義縱向的關系,但並不適合定義橫向的關系,例如日誌功能。日誌代碼往往橫向地散布在所有對象層次中,而與它對應的對象的核心功能毫無關系對於其他類型的代碼,如安全性、異常處理和透明的持續性也都是如此,這種散布在各處的無關的代碼被稱為橫切(cross cutting),在OOP設計中,它導致了大量代碼的重復,而不利於各個模塊的重用。

AOP技術恰恰相反,它利用一種稱為"橫切"的技術,剖解開封裝的對象內部,並將那些影響了多個類的公共行為封裝到一個可重用模塊,並將其命名為"Aspect",即切面。所謂"切面",簡單說就是那些與業務無關,卻為業務模塊所共同調用的邏輯或責任封裝起來,便於減少系統的重復代碼,降低模塊之間的耦合度,並有利於未來的可操作性和可維護性。

使用"橫切"技術,AOP把軟件系統分為兩個部分:核心關註點橫切關註點。業務處理的主要流程是核心關註點,與之關系不大的部分是橫切關註點。橫切關註點的一個特點是,他們經常發生在核心關註點的多處,而各處基本相似,比如權限認證、日誌、事物。AOP的作用在於分離系統中的各種關註點,將核心關註點和橫切關註點分離開來。

AOP核心概念

1、橫切關註點

對哪些方法進行攔截,攔截後怎麽處理,這些關註點稱之為橫切關註點

2、切面(aspect)

類是對物體特征的抽象,切面就是對橫切關註點的抽象

3、連接點(joinpoint)

被攔截到的點,因為Spring只支持方法類型的連接點,所以在Spring中連接點指的就是被攔截到的方法,實際上連接點還可以是字段或者構造器

4、切入點(pointcut)

對連接點進行攔截的定義

5、通知(advice)

所謂通知指的就是指攔截到連接點之後要執行的代碼,通知分為前置、後置、異常、最終、環繞通知五類

6、目標對象

代理的目標對象

7、織入(weave)

將切面應用到目標對象並導致代理對象創建的過程

8、引入(introduction)

在不修改代碼的前提下,引入可以在運行期為類動態地添加一些方法或字段

Spring對AOP的支持

Spring中AOP代理由Spring的IOC容器負責生成、管理,其依賴關系也由IOC容器負責管理。因此,AOP代理可以直接使用容器中的其它bean實例作為目標,這種關系可由IOC容器的依賴註入提供。Spring創建代理的規則為:

1、默認使用Java動態代理來創建AOP代理,這樣就可以為任何接口實例創建代理了

2、當需要代理的類不是代理接口的時候,Spring會切換為使用CGLIB代理,也可強制使用CGLIB

AOP編程其實是很簡單的事情,縱觀AOP編程,程序員只需要參與三個部分:

1、定義普通業務組件

2、定義切入點,一個切入點可能橫切多個業務組件

3、定義增強處理,增強處理就是在AOP框架為普通業務組件織入的處理動作

所以進行AOP編程的關鍵就是定義切入點和定義增強處理,一旦定義了合適的切入點和增強處理,AOP框架將自動生成AOP代理,即:代理對象的方法=增強處理+被代理對象的方法。

基於Spring的AOP簡單實現

註意一下,在講解之前,說明一點:使用Spring AOP,要成功運行起代碼,只用Spring提供給開發者的jar包是不夠的,請額外上網下載兩個jar包:

1、aopalliance.jar

2、aspectjweaver.jar

先定義一個接口

package com.spring;

public interface SpringAop {
    
    void show();
}

然後定義一個實現類

package com.spring;

public class SpringAopIm implements SpringAop {
    
    public void show() {
        
        System.out.println("開始");
        
    }
}

然後是一個橫切類

package com.spring;

public class Minstrel  {
    public void Start() {
        System.out.println("前置通知");
    }
    

    
    public void End() {
        System.out.println("後置通知");
    }
}

最後是一個xml文件

?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:context="http://www.springframework.org/schema/context"
xmlns:aop="http://www.springframework.org/schema/aop" xmlns:tx="http://www.springframework.org/schema/tx"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans-3.0.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context-3.0.xsd
http://www.springframework.org/schema/aop
http://www.springframework.org/schema/aop/spring-aop-3.0.xsd
http://www.springframework.org/schema/tx
http://www.springframework.org/schema/tx/spring-tx-3.0.xsd">
<bean id="minstrel" class="com.spring.Minstrel" /> <!--定義bean 橫切類-->
<bean id="show" class="com.spring.SpringAopIm" /> <!--定義bean 接口實現類-->
<aop:config>
<aop:aspect id="time" ref="minstrel"><!--引用橫切類-->
<aop:pointcut id="addAllMethod" expression="execution(* com.spring.SpringAopIm.show(..))" /><!--橫切的方法 com.spring.SpringAopIm 是類的路徑可以用*代替 show是類中的方法 可以用*代替 (..)是方法的參數-->
<aop:before method="Start" pointcut-ref="addAllMethod" /><!--pointcut-ref 引用被橫切方法的id名 method="Start" 為前置通知的方法名可以自定義 -->
<aop:after method="End" pointcut-ref="addAllMethod" /><!--pointcut-ref 引用被橫切方法的id名 method="Start" 為後置通知的方法名可以自定義-->
</aop:aspect>
</aop:config>
</beans>

然後寫個測試 測試一下

package com.spring;

import org.junit.Test;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;

public class MinstrelTest {
    @Test
    public void show() {
        ApplicationContext context = new ClassPathXmlApplicationContext("spring.xml");//獲取xml
        SpringAop bean = (SpringAop) context.getBean("show");  //獲取bean  bean的名字是你需要橫切的bean id xml中自定義的   由於SpringAopIm實現了SpringAop接口,所以強制轉換必須用父類SpringAop
        bean.show(); //調用方法
    }
}

最後顯示結果

十月 15, 2017 12:08:04 上午 org.springframework.context.support.AbstractApplicationContext prepareRefresh
信息: Refreshing org.springframework.con[email protected]: startup date [Sun Oct 15 00:08:04 CST 2017]; root of context hierarchy
十月 15, 2017 12:08:04 上午 org.springframework.beans.factory.xml.XmlBeanDefinitionReader loadBeanDefinitions
信息: Loading XML bean definitions from class path resource [spring.xml]
十月 15, 2017 12:08:04 上午 org.springframework.beans.factory.support.DefaultListableBeanFactory preInstantiateSingletons
信息: Pre-instantiating singletons in org.s[email protected]ccd26df: defining beans [minstrel,show,org.springframework.aop.config.internalAutoProxyCreator,org.springframework.aop.aspectj.AspectJPointcutAdvisor#0,org.springframework.aop.aspectj.AspectJPointcutAdvisor#1,addAllMethod]; root of factory hierarchy
前置通知
開始
後置通知

註意:

Spring AOP,要成功運行起代碼,只用Spring提供給開發者的jar包是不夠的,請額外上網下載兩個jar包:

1、aopalliance.jar

2、aspectjweaver.jar

引用 http://www.cnblogs.com/hongwz/p/5764917.html 感謝 “獨具匠心”的博客 受益良多

Spring3實戰第一章 Aop 切面 XML配置