剛看spring3實戰書籍第一章 切麵以前沒有關註過 現在看到了 隨手試驗一下 AOP AOP(Aspect Oriented Programming),即面向切麵編程,可以說是OOP(Object Oriented Programming,面向對象編程)的補充和完善。OOP引入封裝、繼承、多態等概 ...
剛看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.context.support.ClassPathXmlApplicationContext@6bb9db06: 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.springframework.beans.factory.support.DefaultListableBeanFactory@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 感謝 “獨具匠心”的博客 受益良多