Spring AOP(Aspect-oriented programming) 是用于切面程式設計,簡單的來說:AOP相當于一個攔截器,去攔截一些處理,例如:當一個方法執行的時候,Spring 能夠攔截正在執行的方法,在方法執行的前或者後增加額外的功能和處理。
在Spring AOP中支援4中類型的通知:
1:before advice 在方法執行前執行。
2:after returning advice 在方法執行後傳回一個結果後執行。
3:after throwing advice 在方法執行過程中抛出異常的時候執行。
4:Around advice 在方法執行前後和抛出異常時執行,相當于綜合了以上三種通知。
下面是一個簡單的AOP advice 的例子:
首先給出一個簡單的Spring 注入的例子,
定義一個Book類:
[java] view plain copy
- package com.myapp.core.aop.advice;
- public class Book {
- private String name;
- private String url;
- private int pages;
- public String getName() {
- return name;
- }
- public void setName(String name) {
- this.name = name;
- }
- public String getUrl() {
- return url;
- }
- public void setUrl(String url) {
- this.url = url;
- }
- public int getPages() {
- return pages;
- }
- public void setPages(int pages) {
- this.pages = pages;
- }
- public void printName(){
- System.out.println("Book name "+ this.name);
- }
- public void printUrl(){
- System.out.println("Book URL "+this.url);
- }
- public void printThrowException(){
- throw new IllegalArgumentException();
- }
- }
相應的配置檔案:
[html] view plain copy
- <?xml version="1.0" encoding="UTF-8"?>
- <beans xmlns="http://www.springframework.org/schema/beans"
- xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
- xsi:schemaLocation="http://www.springframework.org/schema/beans
- http://www.springframework.org/schema/beans/spring-beans-3.0.xsd">
- <!-- more bean definitions for data access objects go here -->
- <bean id="book" class="com.myapp.core.aop.advice.Book">
- <property name="name" value="Effective java" />
- <property name="url" value="www.google.cn"/>
- <property name="pages" value="300" />
- </bean>
- </beans>
對應的測試類:
[java] view plain copy
- package com.myapp.core.aop.advice;
- import org.springframework.context.ApplicationContext;
- import org.springframework.context.support.ClassPathXmlApplicationContext;
- public class MainTest {
- public static void main(String[] args) {
- ApplicationContext context = new ClassPathXmlApplicationContext("resource/aop.xml");
- Book book = (Book) context.getBean("book");
- System.out.println("---------------------");
- book.printName();
- System.out.println("---------------------");
- book.printUrl();
- System.out.println("----------------------");
- try{
- book.printThrowException();
- }catch(Exception e){
- // e.printStackTrace();
- }
- }
- }
輸出結果:
[plain] view plain copy
- 三月 20, 2013 11:01:01 上午 org.springframework.context.support.AbstractApplicationContext prepareRefresh
- INFO: Refreshing org.springframework.context.support.ClassPathXmlApplicationContext@11e831: startup date [Wed Mar 20 11:01:01 CST 2013]; root of context hierarchy
- 三月 20, 2013 11:01:01 上午 org.springframework.beans.factory.xml.XmlBeanDefinitionReader loadBeanDefinitions
- INFO: Loading XML bean definitions from class path resource [resource/aop.xml]
- 三月 20, 2013 11:01:01 上午 org.springframework.beans.factory.support.DefaultListableBeanFactory preInstantiateSingletons
- INFO: Pre-instantiating singletons in org.springframework.beans.factory.support.DefaultListableBeanFac[email protected]: defining beans [book]; root of factory hierarchy
- ---------------------
- Book name Effective java
- ---------------------
- Book URL www.google.cn
- ----------------------
下面對以上的Book加上Spring AOP advices
1:before advice
before advice将在方法執行前執行,建立一個實作MethodBeforeAdvice接口的類能夠定義執行方法前的操作。 類如下: [java] view plain copy
- package com.myapp.core.aop.advice;
- import java.lang.reflect.Method;
- import org.springframework.aop.MethodBeforeAdvice;
- public class BeforeMethod implements MethodBeforeAdvice {
- @Override
- public void before(Method arg0, Object[] arg1, Object arg2)
- throws Throwable {
- // TODO Auto-generated method stub
- System.out.println("Before Method");
- System.out.println("--------------------");
- }
- }
配置對應的bean: 在aop.xml中配置,建立一個BeforeMethod類,一個新的代理命名為:bookProxy 1: target 設定你想攔截的bean 2:interceptorNames設定通知,你想作用于proxy/target上的 對應的配置檔案如下: [html] view plain copy
- <?xml version="1.0" encoding="UTF-8"?>
- <beans xmlns="http://www.springframework.org/schema/beans"
- xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
- xsi:schemaLocation="http://www.springframework.org/schema/beans
- http://www.springframework.org/schema/beans/spring-beans-3.0.xsd">
- <!-- more bean definitions for data access objects go here -->
- <bean id="book" class="com.myapp.core.aop.advice.Book">
- <property name="name" value="Effective java" />
- <property name="url" value="www.google.cn"/>
- <property name="pages" value="300" />
- </bean>
- <bean id="beforeMethodBean" class="com.myapp.core.aop.advice.BeforeMethod" />
- <bean id="bookProxy" class="org.springframework.aop.framework.ProxyFactoryBean">
- <property name="target" ref="book"/>
- <property name="interceptorNames">
- <list>
- <value>beforeMethodBean</value>
- </list>
- </property>
- </bean>
- </beans>
注意:為了使用proxy(代理)我們需要引入 CGLIB2, pom.xml檔案中注入如下: [html] view plain copy
- <dependency>
- <groupId>cglib</groupId>
- <artifactId>cglib</artifactId>
- <version>2.2.2</version>
- </dependency>
運作測試類: [java] view plain copy
- package com.myapp.core.aop.advice;
- import org.springframework.context.ApplicationContext;
- import org.springframework.context.support.ClassPathXmlApplicationContext;
- public class MainTest {
- public static void main(String[] args) {
- ApplicationContext context = new ClassPathXmlApplicationContext("resource/aop.xml");
- Book book = (Book) context.getBean("bookProxy");
- System.out.println("---------------------");
- book.printName();
- System.out.println("---------------------");
- book.printUrl();
- System.out.println("----------------------");
- try{
- book.printThrowException();
- }catch(Exception e){
- // e.printStackTrace();
- }
- }
- }
注意以上獲得的是代理bean; 運作結果如下: [plain] view plain copy
- 三月 20, 2013 2:18:56 下午 org.springframework.context.support.AbstractApplicationContext prepareRefresh
- INFO: Refreshing org.springframework.context.support.ClassPathXmlApplicationContext@eb67e8: startup date [Wed Mar 20 14:18:55 CST 2013]; root of context hierarchy
- 三月 20, 2013 2:18:56 下午 org.springframework.beans.factory.xml.XmlBeanDefinitionReader loadBeanDefinitions
- INFO: Loading XML bean definitions from class path resource [resource/aop.xml]
- 三月 20, 2013 2:18:57 下午 org.springframework.beans.factory.support.DefaultListableBeanFactory preInstantiateSingletons
- INFO: Pre-instantiating singletons in org.springframework.beans.factory.support.DefaultListableBeanFactory@157985: defining beans [book,beforeMethodBean,bookProxy]; root of factory hierarchy
- ---------------------
- Before Method
- --------------------
- Book name Effective java
- ---------------------
- Before Method
- --------------------
- Book URL www.google.cn
- ----------------------
- Before Method
- --------------------
2: after advice
在方法運作傳回結果後将執行這個 afterReturning方法,建立的這個類必須實作:AfterReturningAdvice接口 [java] view plain copy
- package com.myapp.core.aop.advice;
- import java.lang.reflect.Method;
- import org.springframework.aop.AfterReturningAdvice;
- public class AfterMethod implements AfterReturningAdvice {
- @Override
- public void afterReturning(Object arg0, Method arg1, Object[] arg2,
- Object arg3) throws Throwable {
- // TODO Auto-generated method stub
- System.out.println("-------------------");
- System.out.println("After method ");
- }
- }
xml配置檔案: [html] view plain copy
- <?xml version="1.0" encoding="UTF-8"?>
- <beans xmlns="http://www.springframework.org/schema/beans"
- xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
- xsi:schemaLocation="http://www.springframework.org/schema/beans
- http://www.springframework.org/schema/beans/spring-beans-3.0.xsd">
- <!-- more bean definitions for data access objects go here -->
- <bean id="book" class="com.myapp.core.aop.advice.Book">
- <property name="name" value="Effective java" />
- <property name="url" value="www.google.cn"/>
- <property name="pages" value="300" />
- </bean>
- <bean id="beforeMethodBean" class="com.myapp.core.aop.advice.BeforeMethod" />
- <bean id="afterMethodBean" class="com.myapp.core.aop.advice.AfterMethod" />
- <bean id="bookProxy" class="org.springframework.aop.framework.ProxyFactoryBean">
- <property name="target" ref="book"/>
- <property name="interceptorNames">
- <list>
- <value>beforeMethodBean</value>
- <value>afterMethodBean</value>
- </list>
- </property>
- </bean>
- </beans>
運作結果如下: [plain] view plain copy
- 三月 20, 2013 2:22:19 下午 org.springframework.context.support.AbstractApplicationContext prepareRefresh
- INFO: Refreshing org.springframework.context.support.ClassPathXmlApplicationContext@eb67e8: startup date [Wed Mar 20 14:22:19 CST 2013]; root of context hierarchy
- 三月 20, 2013 2:22:19 下午 org.springframework.beans.factory.xml.XmlBeanDefinitionReader loadBeanDefinitions
- INFO: Loading XML bean definitions from class path resource [resource/aop.xml]
- 三月 20, 2013 2:22:20 下午 org.springframework.beans.factory.support.DefaultListableBeanFactory preInstantiateSingletons
- INFO: Pre-instantiating singletons in org.springframework.beans.factory.support.DefaultListableBeanFactory@157985: defining beans [book,beforeMethodBean,afterMethodBean,bookProxy]; root of factory hierarchy
- ---------------------
- Before Method
- --------------------
- Book name Effective java
- -------------------
- After method
- ---------------------
- Before Method
- --------------------
- Book URL www.google.cn
- -------------------
- After method
- ----------------------
- Before Method
- --------------------
3:after throwing advice
當方法執行抛出一個異常後,會執行這個方法,建立一個類實作:ThrowsAdvice接口,建立一個afterThrowing攔截:IllegalArgumentException異常。 類如下: [java] view plain copy
- package com.myapp.core.aop.advice;
- import org.springframework.aop.ThrowsAdvice;
- public class ThrowException implements ThrowsAdvice{
- public void afterThrowing(IllegalArgumentException e) throws Throwable{
- System.out.println("after Throwing Exception");
- }
- }
xml中配置檔案如下: [html] view plain copy
- <?xml version="1.0" encoding="UTF-8"?>
- <beans xmlns="http://www.springframework.org/schema/beans"
- xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
- xsi:schemaLocation="http://www.springframework.org/schema/beans
- http://www.springframework.org/schema/beans/spring-beans-3.0.xsd">
- <!-- more bean definitions for data access objects go here -->
- <bean id="book" class="com.myapp.core.aop.advice.Book">
- <property name="name" value="Effective java" />
- <property name="url" value="www.google.cn"/>
- <property name="pages" value="300" />
- </bean>
- <!-- before advice -->
- <bean id="beforeMethodBean" class="com.myapp.core.aop.advice.BeforeMethod" />
- <!-- after advice -->
- <bean id="afterMethodBean" class="com.myapp.core.aop.advice.AfterMethod" />
- <!-- throwing advice -->
- <bean id="throwException" class="com.myapp.core.aop.advice.ThrowException" />
- <bean id="bookProxy" class="org.springframework.aop.framework.ProxyFactoryBean">
- <property name="target" ref="book"/>
- <property name="interceptorNames">
- <list>
- <value>beforeMethodBean</value>
- <value>afterMethodBean</value>
- <value>throwException</value>
- </list>
- </property>
- </bean>
- </beans>
執行結果如下: [plain] view plain copy
- 三月 20, 2013 2:37:36 下午 org.springframework.context.support.AbstractApplicationContext prepareRefresh
- INFO: Refreshing org.springframework.context.support.ClassPathXmlApplicationContext@eb67e8: startup date [Wed Mar 20 14:37:36 CST 2013]; root of context hierarchy
- 三月 20, 2013 2:37:36 下午 org.springframework.beans.factory.xml.XmlBeanDefinitionReader loadBeanDefinitions
- INFO: Loading XML bean definitions from class path resource [resource/aop.xml]
- 三月 20, 2013 2:37:36 下午 org.springframework.beans.factory.support.DefaultListableBeanFactory preInstantiateSingletons
- INFO: Pre-instantiating singletons in org.springframework.beans.factory.support.DefaultListableBeanFactory@157985: defining beans [book,beforeMethodBean,afterMethodBean,throwException,bookProxy]; root of factory hierarchy
- ---------------------
- Before Method
- --------------------
- Book name Effective java
- -------------------
- After method
- ---------------------
- Before Method
- --------------------
- Book URL www.google.cn
- -------------------
- After method
- ----------------------
- Before Method
- --------------------
- after Throwing Exception
4:Around advice
這個advice 聯合了上面的三個advices,在方法執行期間執行,建立一個類實作MethodInterceptor接口,需要在方法中執行Object result = methodInvocation.proceed();方法才能得到執行,否則方法不會執行。 類如下: [java] view plain copy
- package com.myapp.core.aop.advice;
- import java.util.Arrays;
- import org.aopalliance.intercept.MethodInterceptor;
- import org.aopalliance.intercept.MethodInvocation;
- public class AroundMethod implements MethodInterceptor{
- @Override
- public Object invoke(MethodInvocation methodInvocation) throws Throwable {
- // TODO Auto-generated method stub
- System.out.println("method name:" + methodInvocation.getMethod().getName());
- System.out.println("method arguments" + Arrays.toString(methodInvocation.getArguments()));
- System.out.println("Around method : before ");
- try{
- Object result = methodInvocation.proceed();
- System.out.println("Around method : after ");
- return result;
- }catch(IllegalArgumentException e){
- System.out.println("Around method : throw an exception ");
- throw e;
- }
- }
- }
配置檔案如下: [html] view plain copy
- <?xml version="1.0" encoding="UTF-8"?>
- <beans xmlns="http://www.springframework.org/schema/beans"
- xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
- xsi:schemaLocation="http://www.springframework.org/schema/beans
- http://www.springframework.org/schema/beans/spring-beans-3.0.xsd">
- <!-- more bean definitions for data access objects go here -->
- <bean id="book" class="com.myapp.core.aop.advice.Book">
- <property name="name" value="Effective java" />
- <property name="url" value="www.google.cn"/>
- <property name="pages" value="300" />
- </bean>
- <bean id="aroundMethod" class="com.myapp.core.aop.advice.AroundMethod" />
- <bean id="bookProxy" class="org.springframework.aop.framework.ProxyFactoryBean" >
- <property name="target" ref="book"/>
- <property name="interceptorNames">
- <list>
- <value>aroundMethod</value>
- </list>
- </property>
- </bean>
- </beans>
測試結果: [plain] view plain copy
- 三月 20, 2013 3:02:19 下午 org.springframework.context.support.AbstractApplicationContext prepareRefresh
- INFO: Refreshing org.springframework.context.support.ClassPathXmlApplicationContext@eb67e8: startup date [Wed Mar 20 15:02:19 CST 2013]; root of context hierarchy
- 三月 20, 2013 3:02:19 下午 org.springframework.beans.factory.xml.XmlBeanDefinitionReader loadBeanDefinitions
- INFO: Loading XML bean definitions from class path resource [resource/aop.xml]
- 三月 20, 2013 3:02:19 下午 org.springframework.beans.factory.support.DefaultListableBeanFactory preInstantiateSingletons
- INFO: Pre-instantiating singletons in org.springframework.beans.factory.support.DefaultListableBeanFactory@e29127: defining beans [book,aroundMethod,bookProxy]; root of factory hierarchy
- ---------------------
- method name:printName
- method arguments[]
- Around method : before
- Book name Effective java
- Around method : after
- ---------------------
- method name:printUrl
- method arguments[]
- Around method : before
- Book URL www.google.cn
- Around method : after
- ----------------------
- method name:printThrowException
- method arguments[]
- Around method : before
- Around method : throw an exception
around advice得到實作。 over