# 【第六章】 AOP 之 6.5 AspectJ切入點語法詳解 ——跟我學spring3
### 6.5.1? Spring AOP支持的AspectJ切入點指示符
切入點指示符用來指示切入點表達式目的,,在Spring AOP中目前只有執行方法這一個連接點,Spring AOP支持的AspectJ切入點指示符如下:
**execution:**用于匹配方法執行的連接點;
_**within**__**:**__用于匹配指定類型內的方法執行;_
_**this**__**:**__用于匹配當前__AOP__代理對象類型的執行方法;注意是__AOP__代理對象的類型匹配,這樣就可能包括引入接口也類型匹配;_
_**target**__**:**__用于匹配當前目標對象類型的執行方法;注意是目標對象的類型匹配,這樣就不包括引入接口也類型匹配;_
_**args**__**:**__用于匹配當前執行的方法傳入的參數為指定類型的執行方法;_
_**@within**__**:**__用于匹配所以持有指定注解類型內的方法__;_
_**@target**__**:**__用于匹配當前目標對象類型的執行方法,其中目標對象持有指定的注解;_
_**@args**__**:**__用于匹配當前執行的方法傳入的參數持有指定注解的執行;_
_**@annotation**__**:**__用于匹配當前執行方法持有指定注解的方法;_
_**bean**__**:**__Spring AOP__擴展的,__AspectJ__沒有對于指示符,用于匹配特定名稱的__Bean__對象的執行方法;_
_**reference pointcut**__**:**__表示引用其他命名切入點,只有__@ApectJ__風格支持,__Schema__風格不支持。_
AspectJ切入點支持的切入點指示符還有: call、get、set、preinitialization、staticinitialization、initialization、handler、adviceexecution、withincode、cflow、cflowbelow、if、@this、@withincode;但Spring AOP目前不支持這些指示符,使用這些指示符將拋出IllegalArgumentException異常。這些指示符Spring AOP可能會在以后進行擴展。
### 6.5.1? 命名及匿名切入點
命名切入點可以被其他切入點引用,而匿名切入點是不可以的。
只有@AspectJ支持命名切入點,而Schema風格不支持命名切入點。
如下所示,@AspectJ使用如下方式引用命名切入點:
### 
### 6.5.2??????? ;類型匹配語法
首先讓我們來了解下AspectJ類型匹配的通配符:
* :匹配任何數量字符;
.. :(兩個點)匹配任何數量字符的重復,如在類型模式中匹配任何數量子包;而在方法參數模式中匹配任何數量參數。
**+?:**匹配指定類型的子類型;僅能作為后綴放在類型模式后邊。
> **java.lang.String?**???匹配String類型;
>
> **java.*.String**? ? ? ? ?匹配java包下的任何“一級子包”下的String類型;
>
> ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? 如匹配java.lang.String,但不匹配java.lang.ss.String
>
> **java..*?**? ? ? ? ? ? ? ? ?匹配java包及任何子包下的任何類型;
>
> ??????????????????????????? ? 如匹配java.lang.String、java.lang.annotation.Annotation
>
> **java.lang.*ing??**????匹配任何java.lang包下的以ing結尾的類型;
>
> **java.lang.Number+**? 匹配java.lang包下的任何Number的自類型;
>
> ??????????????????????????? ?? 如匹配java.lang.Integer,也匹配java.math.BigInteger
接下來再看一下具體的匹配表達式類型吧:
**匹配類型:**使用如下方式匹配
```
注解? 類的全限定名字
```
* **注解:**可選,類型上持有的注解,如@Deprecated;
* **類的全限定名:**必填,可以是任何類全限定名。
**匹配方法執行:**使用如下方式匹配:
```
注解? 修飾符? 返回值類型 類型聲明?方法名(參數列表) 異常列表?
```
* **注解:**可選,方法上持有的注解,如@Deprecated;
* **修飾符:**可選,如public、protected;
* **返回值類型:**必填,可以是任何類型模式;“*”表示所有類型;
* **類型聲明:**可選,可以是任何類型模式;
* **方法名:**必填,可以使用“*”進行模式匹配;
* **參數列表:**“()”表示方法沒有任何參數;“(..)”表示匹配接受任意個參數的方法,“(..,java.lang.String)”表示匹配接受java.lang.String類型的參數結束,且其前邊可以接受有任意個參數的方法;“(java.lang.String,..)” 表示匹配接受java.lang.String類型的參數開始,且其后邊可以接受任意個參數的方法;“(*,java.lang.String)” 表示匹配接受java.lang.String類型的參數結束,且其前邊接受有一個任意類型參數的方法;
* **異常列表:**可選,以“throws 異常全限定名列表”聲明,異常全限定名列表如有多個以“,”分割,如throws java.lang.IllegalArgumentException, java.lang.ArrayIndexOutOfBoundsException。
**匹配Bean名稱:**可以使用Bean的id或name進行匹配,并且可使用通配符“*”;
### 6.5.3? 組合切入點表達式
AspectJ使用 且(&&)、或(||)、非(!)來組合切入點表達式。
在Schema風格下,由于在XML中使用“&&”需要使用轉義字符“&&”來代替之,所以很不方便,因此Spring ASP 提供了and、or、not來代替&&、||、!。
### 6.5.4? 切入點使用示例
**一、execution:**使用“execution(方法表達式)”匹配方法執行;
| **模式** | **描述** |
| --- | --- |
| public * *(..) | 任何公共方法的執行 |
| * cn.javass..IPointcutService.*() | cn.javass包及所有子包下IPointcutService接口中的任何無參方法 |
| * cn.javass..*.*(..) | cn.javass包及所有子包下任何類的任何方法 |
| * cn.javass..IPointcutService.*(*) | cn.javass包及所有子包下IPointcutService接口的任何只有一個參數方法 |
| * (!cn.javass..IPointcutService+).*(..) | 非“cn.javass包及所有子包下IPointcutService接口及子類型”的任何方法 |
| * cn.javass..IPointcutService+.*() | cn.javass包及所有子包下IPointcutService接口及子類型的的任何無參方法 |
| * cn.javass..IPointcut*.test*(java.util.Date) | cn.javass包及所有子包下IPointcut前綴類型的的以test開頭的只有一個參數類型為java.util.Date的方法,注意該匹配是根據方法簽名的參數類型進行匹配的,而不是根據執行時傳入的參數類型決定的如定義方法:public void test(Object obj);即使執行時傳入java.util.Date,也不會匹配的; |
| * cn.javass..IPointcut*.test*(..) ?throws?IllegalArgumentException, ArrayIndexOutOfBoundsException | cn.javass包及所有子包下IPointcut前綴類型的的任何方法,且拋出IllegalArgumentException和ArrayIndexOutOfBoundsException異常 |
| * (cn.javass..IPointcutService+&& java.io.Serializable+).*(..) | 任何實現了cn.javass包及所有子包下IPointcutService接口和java.io.Serializable接口的類型的任何方法 |
| @java.lang.Deprecated * *(..) | 任何持有@java.lang.Deprecated注解的方法 |
| @java.lang.Deprecated @cn.javass..Secure ?* *(..) | 任何持有@java.lang.Deprecated和@cn.javass..Secure注解的方法 |
| @(java.lang.Deprecated || cn.javass..Secure) * *(..) | 任何持有@java.lang.Deprecated或@ cn.javass..Secure注解的方法 |
| (@cn.javass..Secure ?*)? *(..) | 任何返回值類型持有@cn.javass..Secure的方法 |
| * ?(@cn.javass..Secure *).*(..) | 任何定義方法的類型持有@cn.javass..Secure的方法 |
| * *(@cn.javass..Secure (*) , @cn.javass..Secure (*)) | 任何簽名帶有兩個參數的方法,且這個兩個參數都被@ Secure標記了,如public void test(@Secure String str1,?@Secure String str1); |
| * *((@ cn.javass..Secure *))或* *(@ cn.javass..Secure *) | 任何帶有一個參數的方法,且該參數類型持有@ cn.javass..Secure;如public void test(Model model);且Model類上持有@Secure注解 |
| * *(@cn.javass..Secure (@cn.javass..Secure *) ,@ cn.javass..Secure (@cn.javass..Secure *)) | 任何帶有兩個參數的方法,且這兩個參數都被@ cn.javass..Secure標記了;且這兩個參數的類型上都持有@ cn.javass..Secure; |
| * *(java.util.Map<cn.javass..Model, cn.javass..Model>, ..) | 任何帶有一個java.util.Map參數的方法,且該參數類型是以< cn.javass..Model, cn.javass..Model >為泛型參數;注意只匹配第一個參數為java.util.Map,不包括子類型;如public void test(HashMap<Model, Model> map, String str);將不匹配,必須使用“* *(java.util.HashMap<cn.javass..Model,cn.javass..Model>, ..)”進行匹配;而public void test(Map map, int i);也將不匹配,因為泛型參數不匹配 |
| * *(java.util.Collection<@cn.javass..Secure *>) | 任何帶有一個參數(類型為java.util.Collection)的方法,且該參數類型是有一個泛型參數,該泛型參數類型上持有@cn.javass..Secure注解;如public void test(Collection<Model> collection);Model類型上持有@cn.javass..Secure |
| * *(java.util.Set<? extends HashMap>) | 任何帶有一個參數的方法,且傳入的參數類型是有一個泛型參數,該泛型參數類型繼承與HashMap;**Spring AOP目前測試不能正常工作** |
| * *(java.util.List<? super HashMap>) | 任何帶有一個參數的方法,且傳入的參數類型是有一個泛型參數,該泛型參數類型是HashMap的基類型;如public voi test(Map map);**Spring AOP目前測試不能正常工作** |
| * *(*<@cn.javass..Secure *>) | 任何帶有一個參數的方法,且該參數類型是有一個泛型參數,該泛型參數類型上持有@cn.javass..Secure注解;**Spring AOP目前測試不能正常工作** |
**二**_、_**within:使用“within(類型表達式)”**_匹配指定類型內的方法執行;_
| **模式** | **描述** |
| --- | --- |
| within(cn.javass..*) | cn.javass包及子包下的任何方法執行 |
| within(cn.javass..IPointcutService+) | cn.javass包或所有子包下IPointcutService類型及子類型的任何方法 |
| within(@cn.javass..Secure *) | 持有cn.javass..Secure注解的任何類型的任何方法必須是在目標對象上聲明這個注解,在接口上聲明的對它不起作用 |
**三**_、__**this**__**:**__使用“__this(__類型全限定名__)__”匹配當前__AOP__代理對象類型的執行方法;注意是__AOP__代理對象的類型匹配,這樣就可能包括引入接口方法也可以匹配;注意__this__中使用的表達式必須是類型全限定名,不支持通配符;_
| **模式** | **描述** |
| --- | --- |
| this(cn.javass.spring.chapter6.service.IPointcutService) | 當前AOP對象實現了 IPointcutService接口的任何方法 |
| this(cn.javass.spring.chapter6.service.IIntroductionService) | 當前AOP對象實現了 IIntroductionService接口的任何方法也可能是引入接口 |
_四、__**target**__**:**__使用“__target(__類型全限定名__)__”匹配當前目標對象類型的執行方法;注意是目標對象的類型匹配,這樣就不包括引入接口也類型匹配;注意__target__中使用的表達式必須是類型全限定名,不支持通配符;_
| **模式** | **描述** |
| --- | --- |
| target(cn.javass.spring.chapter6.service.IPointcutService) | 當前目標對象(非AOP對象)實現了 IPointcutService接口的任何方法 |
| target(cn.javass.spring.chapter6.service.IIntroductionService) | 當前目標對象(非AOP對象) 實現了IIntroductionService 接口的任何方法不可能是引入接口 |
_五、__**args**__**:**__使用“__args(__參數類型列表__)__”匹配當前執行的方法傳入的參數為指定類型的執行方法;注意是匹配傳入的參數類型,不是匹配方法簽名的參數類型;參數類型列表中的參數必須是類型全限定名,通配符不支持;__args__屬于動態切入點,這種切入點開銷非常大,非特殊情況最好不要使用;_
| **模式** | **描述** |
| --- | --- |
| args (java.io.Serializable,..) | 任何一個以接受“傳入參數類型為 java.io.Serializable” 開頭,且其后可跟任意個任意類型的參數的方法執行,args指定的參數類型是在運行時動態匹配的 |
_**六、**__**@within**__**:**__使用“__@within(__注解類型__)__”匹配所以持有指定注解類型內的方法;注解類型也必須是全限定類型名;_
| **模式** | **描述** |
| --- | --- |
| @within cn.javass.spring.chapter6.Secure) | 任何目標對象對應的類型持有Secure注解的類方法;必須是在目標對象上聲明這個注解,在接口上聲明的對它不起作用 |
_**七、**__**@target**__**:**__使用“__@target(__注解類型__)__”匹配當前目標對象類型的執行方法,其中目標對象持有指定的注解;注解類型也必須是全限定類型名;_
| **模式** | **描述** |
| --- | --- |
| @target (cn.javass.spring.chapter6.Secure) | 任何目標對象持有Secure注解的類方法;必須是在目標對象上聲明這個注解,在接口上聲明的對它不起作用 |
_**八、**__**@args**__**:**__使用“__@args(__注解列表__)__”匹配當前執行的方法傳入的參數持有指定注解的執行;注解類型也必須是全限定類型名;_
| **模式** | **描述** |
| --- | --- |
| @args (cn.javass.spring.chapter6.Secure) | 任何一個只接受一個參數的方法,且方法運行時傳入的參數持有注解 cn.javass.spring.chapter6.Secure;動態切入點,類似于arg指示符; |
_**九、**__**@annotation**__**:**__使用“__@annotation(__注解類型__)__”匹配當前執行方法持有指定注解的方法;注解類型也必須是全限定類型名;_
| **模式** | **描述** |
| --- | --- |
| @annotation(cn.javass.spring.chapter6.Secure ) | 當前執行方法上持有注解 cn.javass.spring.chapter6.Secure將被匹配 |
_**十、**__**bean**__**:**__使用“__bean(Bean id__或名字通配符__)__”__匹配特定名稱的__Bean__對象的執行方法;__Spring ASP__擴展的,在__AspectJ__中無相應概念;_
| **模式** | **描述** |
| --- | --- |
| bean(*Service) | 匹配所有以Service命名(id或name)結尾的Bean |
**十一、reference pointcut:**表示引用其他命名切入點,只有@ApectJ風格支持,Schema風格不支持,如下所示:

比如我們定義如下切面:
```
package cn.javass.spring.chapter6.aop;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Pointcut;
@Aspect
public class ReferencePointcutAspect {
@Pointcut(value="execution(* *())")
public void pointcut() {}
}
```
可以通過如下方式引用:
```
@Before(value = "cn.javass.spring.chapter6.aop.ReferencePointcutAspect.pointcut()")
public void referencePointcutTest2(JoinPoint jp) {}
```
除了可以在@AspectJ風格的切面內引用外,也可以在Schema風格的切面定義內引用,引用方式與@AspectJ完全一樣。
到此我們切入點表達式語法示例就介紹完了,我們這些示例幾乎包含了日常開發中的所有情況,但當然還有更復雜的語法等等,如果以上介紹的不能滿足您的需要,請參考AspectJ文檔。
由于測試代碼相當長,所以為了節約篇幅本示例代碼在cn.javass.spring.chapter6\. PointcutTest文件中,需要時請參考該文件。
原創內容,轉自請注明出處【[http://sishuok.com/forum/blogPost/list/0/2472.html](http://sishuok.com/forum/blogPost/list/0/2472.html#7144)】
- 跟我學 Spring3
- 【第二章】 IoC 之 2.1 IoC基礎 ——跟我學Spring3
- 【第二章】 IoC 之 2.2 IoC 容器基本原理 ——跟我學Spring3
- 【第二章】 IoC 之 2.3 IoC的配置使用——跟我學Spring3
- 【第三章】 DI 之 3.1 DI的配置使用 ——跟我學spring3
- 【第三章】 DI 之 3.2 循環依賴 ——跟我學spring3
- 【第三章】 DI 之 3.3 更多DI的知識 ——跟我學spring3
- 【第三章】 DI 之 3.4 Bean的作用域 ——跟我學spring3
- 【第四章】 資源 之 4.1 基礎知識 ——跟我學spring3
- 【第四章】 資源 之 4.2 內置Resource實現 ——跟我學spring3
- 【第四章】 資源 之 4.3 訪問Resource ——跟我學spring3
- 【第四章】 資源 之 4.4 Resource通配符路徑 ——跟我學spring3
- 【第五章】Spring表達式語言 之 5.1 概述 5.2 SpEL基礎 ——跟我學spring3
- 【第五章】Spring表達式語言 之 5.3 SpEL語法 ——跟我學spring3
- 【第五章】Spring表達式語言 之 5.4在Bean定義中使用EL—跟我學spring3
- 【第六章】 AOP 之 6.1 AOP基礎 ——跟我學spring3
- 【第六章】 AOP 之 6.2 AOP的HelloWorld ——跟我學spring3
- 【第六章】 AOP 之 6.3 基于Schema的AOP ——跟我學spring3
- 【第六章】 AOP 之 6.4 基于@AspectJ的AOP ——跟我學spring3
- 【第六章】 AOP 之 6.5 AspectJ切入點語法詳解 ——跟我學spring3
- 【第六章】 AOP 之 6.6 通知參數 ——跟我學spring3
- 【第六章】 AOP 之 6.7 通知順序 ——跟我學spring3
- 【第六章】 AOP 之 6.8 切面實例化模型 ——跟我學spring3
- 【第六章】 AOP 之 6.9 代理機制 ——跟我學spring3
- 【第七章】 對JDBC的支持 之 7.1 概述 ——跟我學spring3
- 【第七章】 對JDBC的支持 之 7.2 JDBC模板類 ——跟我學spring3
- 【第七章】 對JDBC的支持 之 7.3 關系數據庫操作對象化 ——跟我學spring3
- 【第七章】 對JDBC的支持 之 7.4 Spring提供的其它幫助 ——跟我學spring3【私塾在線原創】
- 【第七章】 對JDBC的支持 之 7.5 集成Spring JDBC及最佳實踐 ——跟我學spring3
- 【第八章】 對ORM的支持 之 8.1 概述 ——跟我學spring3
- 【第八章】 對ORM的支持 之 8.2 集成Hibernate3 ——跟我學spring3
- 【第八章】 對ORM的支持 之 8.3 集成iBATIS ——跟我學spring3
- 【第八章】 對ORM的支持 之 8.4 集成JPA ——跟我學spring3
- 【第九章】 Spring的事務 之 9.1 數據庫事務概述 ——跟我學spring3
- 【第九章】 Spring的事務 之 9.2 事務管理器 ——跟我學spring3
- 【第九章】 Spring的事務 之 9.3 編程式事務 ——跟我學spring3
- 【第九章】 Spring的事務 之 9.4 聲明式事務 ——跟我學spring3
- 【第十章】集成其它Web框架 之 10.1 概述 ——跟我學spring3
- 【第十章】集成其它Web框架 之 10.2 集成Struts1.x ——跟我學spring3
- 【第十章】集成其它Web框架 之 10.3 集成Struts2.x ——跟我學spring3
- 【第十章】集成其它Web框架 之 10.4 集成JSF ——跟我學spring3
- 【第十一章】 SSH集成開發積分商城 之 11.1 概述 ——跟我學spring3
- 【第十一章】 SSH集成開發積分商城 之 11.2 實現通用層 ——跟我學spring3
- 【第十一章】 SSH集成開發積分商城 之 11.3 實現積分商城層 ——跟我學spring3
- 【第十二章】零配置 之 12.1 概述 ——跟我學spring3
- 【第十二章】零配置 之 12.2 注解實現Bean依賴注入 ——跟我學spring3
- 【第十二章】零配置 之 12.3 注解實現Bean定義 ——跟我學spring3
- 【第十二章】零配置 之 12.4 基于Java類定義Bean配置元數據 ——跟我學spring3
- 【第十二章】零配置 之 12.5 綜合示例-積分商城 ——跟我學spring3
- 【第十三章】 測試 之 13.1 概述 13.2 單元測試 ——跟我學spring3
- 【第十三章】 測試 之 13.3 集成測試 ——跟我學spring3
- 跟我學 Spring MVC
- SpringMVC + spring3.1.1 + hibernate4.1.0 集成及常見問題總結
- Spring Web MVC中的頁面緩存支持 ——跟我學SpringMVC系列
- Spring3 Web MVC下的數據類型轉換(第一篇)——《跟我學Spring3 Web MVC》搶先看
- Spring3 Web MVC下的數據格式化(第二篇)——《跟我學Spring3 Web MVC》搶先看
- 第一章 Web MVC簡介 —— 跟開濤學SpringMVC
- 第二章 Spring MVC入門 —— 跟開濤學SpringMVC
- 第三章 DispatcherServlet詳解 ——跟開濤學SpringMVC
- 第四章 Controller接口控制器詳解(1)——跟著開濤學SpringMVC
- 第四章 Controller接口控制器詳解(2)——跟著開濤學SpringMVC
- 第四章 Controller接口控制器詳解(3)——跟著開濤學SpringMVC
- 第四章 Controller接口控制器詳解 (4)——跟著開濤學SpringMVC
- 第四章 Controller接口控制器詳解(5)——跟著開濤學SpringMVC
- 跟著開濤學SpringMVC 第一章源代碼下載
- 第二章 Spring MVC入門 源代碼下載
- 第四章 Controller接口控制器詳解 源代碼下載
- 第四章 Controller接口控制器詳解(6)——跟著開濤學SpringMVC
- 第四章 Controller接口控制器詳解(7 完)——跟著開濤學SpringMVC
- 第五章 處理器攔截器詳解——跟著開濤學SpringMVC
- 源代碼下載 第五章 處理器攔截器詳解——跟著開濤學SpringMVC
- 注解式控制器運行流程及處理器定義 第六章 注解式控制器詳解——跟著開濤學SpringMVC
- 源代碼下載 第六章 注解式控制器詳解
- SpringMVC3強大的請求映射規則詳解 第六章 注解式控制器詳解——跟著開濤學SpringMVC
- Spring MVC 3.1新特性 生產者、消費者請求限定 —— 第六章 注解式控制器詳解——跟著開濤學SpringMVC
- SpringMVC強大的數據綁定(1)——第六章 注解式控制器詳解——跟著開濤學SpringMVC
- SpringMVC強大的數據綁定(2)——第六章 注解式控制器詳解——跟著開濤學SpringMVC
- SpringMVC數據類型轉換——第七章 注解式控制器的數據驗證、類型轉換及格式化——跟著開濤學SpringMVC
- SpringMVC數據格式化——第七章 注解式控制器的數據驗證、類型轉換及格式化——跟著開濤學SpringMVC
- SpringMVC數據驗證——第七章 注解式控制器的數據驗證、類型轉換及格式化——跟著開濤學SpringMVC