Spring的基本用法(大全-AOP)
Spring的AOP
面向切面程式設計(Aspect Orient Programming)分成兩類:
- 靜態AOP實現:AOP框架在編譯階段對程式進行修改,即實現對目標類的增強,生成靜態的AOP代理類。以Aspect為代表
- 動態AOP實現:AOP框架在執行階段動態生成AOP代理,即實現對目標物件的增強。以Spring AOP為代表
AspectJ是基於Java語言的AOP框架。
- 切面(Aspect):切面用於組織多個Advice,Advice放在切面中定義
- 連線點(Joinpoint):程式執行過程中明確點,如方法的呼叫,或者異常的丟擲。在Spring AOP中,連線點總是方法的呼叫
- 增強處理(Advice):AOP框架在特定的切入點執行的增強處理。處理有“around”、“before”、“after”
- 切入點(Pointcut):可以插入增強處理的連線點。當某個連線點滿足指定要求時,該連線點將被新增增強處理,該連線點也就變成切入點。連線點+增強處理==切入點。包含切入點表示式和名字、任意引數的方法的簽名。
- AspectJ的切入點表示式:@Pointcut註解來標識
- 引入:將方法或欄位使用新增到被處理的類中。允許將新的介面引入到任何被處理的物件中去。
- 目標物件:被AOP框架增強處理的物件。被增強的物件。AOP框架採用動態AOP實現,該物件是被代理物件。
- 織入:將增強處理新增到目標物件中,並建立一個被增強的物件(AOP代理)的過程。
AOP代理就是AOP框架動態生成一個物件。該物件被作為目標物件。AOP代理包含目標物件的全部方法。AOP方法在特定切入點添加了增強處理,回撥目標物件的方法。
Spring僅支援將方法呼叫作為連線點。如果需要對成員變數的訪問和更新作為增強處理的連線點,考慮使用AspectJ。
- 定義普通業務元件
- 定義切入點,一個切入點可能橫切多個業務元件
- 定義增強處理,增強處理就是在AOP框架為普通業務元件織入的處理動作。
AOP代理的方法 = 增強處理 + 目標物件的方法
零配置方式
@Aspect標識切面
@Pointcut標記切入點
不打算使用Spring的XMl Schema配置方式。
<!-- 啟動@AspectJ支援 -->
<bean class="org.springframeworking.aop.aspectj.annotation.AnnotationAwareAspectJAutoProxyCreator" />
- 定義切面Bean。新增@Aspect註解為切面類,不會對該bean類進行增強
- 定義增強處理。型別如下:
Before增強處理
在切面類中用@Before("excution()")修飾一個方法,需要指定value屬性值。該屬性值為切入點表示式(如excution(org.aaa.bbb.*.*(..)))。用於指定增強處理織入哪些切入點。
我的理解:用@Aspect修飾一個類為切面類,在用@Before("excution(org.aaa.bbb.*.*(..))")修飾類中的方法,在bbb包下所有的類的方法作為切入點(所有方法中的任意一個方法執行之前都要先呼叫一下用@before修飾的這個方法)
定義AfterReturning增強處理
目標方法完成後織入。
@AfterReturning註解的屬性:
pointcut/value:指定對應的切入點表示式
returning:指定一個形參名。訪問目標方法的返回值,表示Advice可以定義與此相同的形參。限制目標方法返回指定型別的值或沒有返回值。
AfterThrowing增強處理
處理程式未處理的異常,@AfterThrowing的屬性如下
pointcut/value:指定對應的切入點表示式
throwing:指定一個形參名。該形參可用於訪問目標方法丟擲的異常。限制目標方法必須丟擲指定型別的異常。
After增強處理
無論目標方法成功完成還是遇到異常終止,他都會被織入。所以After必須準備處理正常返回和異常返回兩種情況,常用於釋放資源。
@after("excution()")
Around增強處理
before增強處理+AfterReturning增強處理。不同的是它可以決定目標方法在什麼時候執行,是如何執行,甚至可以阻止目標方法的執行
既可以在目標方法執行之前織入,也可以在目標方法執行之後織入增強動作。
可以改變執行目標方法的引數值和返回值。
切入點表示式
切入點指示符有如下:
- excution,屬性有modifiers-pattern(指定方法的修飾符)、ret-type-pattern(指定方法的返回值)等
- within:用於限定匹配特定型別的連線點。Spring AOP使用時,只能匹配方法執行的連線點。
- this:用於限定AOP代理的必須是指定型別的例項,匹配該物件的所有連線點。Spring AOP使用時,只能匹配方法執行的連線點。
- target:限定目標物件必須是指定型別的例項。Spring AOP使用時,只能匹配方法執行的連線點。
- args:用於對連線點的引數進行限制,要求引數是指定型別的例項。Spring AOP使用時,只能匹配方法執行的連線點。
XML配置檔案方式
自動掃描Bean元件和切面類<context:component-scan base-package=" "><context:include-filter type=" " expression="" /></context:component-scan>
<aop:aspectj-autoproxy/>顯示啟動自動代理。要麼全部使用自動代理方式,要麼使用<aop:config/>
注意:所有的切面、切入點和增強處理都要包含在<aop:config/>元素內部。<beans/>元素下可以包含多個<aop:config/>。
1.配置切面
<aop:aspect>的屬性:
- id:切面的標誌名
- ref:將ref所引用的普通Bean轉換為切面Bean
- order:定義該切面Bean的優先順序。與@AspectJ的@Order註解作用一樣
<aop:pointcut>的屬性:
id:切入點的標誌名
expression:切入點表示式
2.配置增強處理
<aop:before/>、<aop:after/>、<aop:afterreturning/>、<aop:afterthrowing/>、<aop:around/>
屬性:
pointcut:切入表示式
pointcut-ref:指定一個已存在的切入點名稱
method:指定方法名
throwing:指定形參名,<after-throwing/>有效
returning:指定形參名,<after-returning/>有效
Spring的事務
全域性事務和區域性事務
全域性事務由應用伺服器管理,需要JTA(Java Transaction API)支援
區域性事務和底層採用持久化技術。採用JDBC持久化,需要使用Connection物件來操作事務;採用Hibernate持久化技術需要使用Session物件操作事務。
mybatis事務有兩種使用方式:
(a):使用JDBC的事務管理機制:即使用java.Sql.Connection物件完成對事務的提交,回滾和關閉操作。
(b):使用MANAGED的事務管理機制:mybatis本身不會去實現事務管理的相關操作,而是交個外部容器(JBOSS,WebLogic)來管理事務。當與spring整合使用後,一般使用spring來管理事務。
宣告式事務管理建立在AOP之上的。其本質是對方法前後進行攔截,然後在目標方法開始之前建立或者織入一個事務,在執行完目標方法之後根據執行情況提交或者回滾事務。宣告式事務最大的優點就是不需要通過程式設計的方式管理事務,這樣就不需要在業務邏輯程式碼中摻雜事務管理的程式碼,只需在配置檔案中做相關的事務規則宣告(或通過基於@Transactional註解的方式),便可以將事務規則應用到業務邏輯中。
事務規則的屬性值:事務隔離、事務傳播、事務超時、只讀狀態。
事務傳播行為
@Transactional所謂事務的傳播行為是指,如果在開始當前事務之前,一個事務上下文已經存在,此時有若干選項可以指定一個事務性方法的執行行為。在TransactionDefinition定義中包括瞭如下幾個表示傳播行為的常量:
- TransactionDefinition.PROPAGATION_REQUIRED:如果當前存在事務,則加入該事務;如果當前沒有事務,則建立一個新的事務。這是預設值。
- TransactionDefinition.PROPAGATION_REQUIRES_NEW:建立一個新的事務,如果當前存在事務,則把當前事務掛起。
- TransactionDefinition.PROPAGATION_SUPPORTS:如果當前存在事務,則加入該事務;如果當前沒有事務,則以非事務的方式繼續執行。
- TransactionDefinition.PROPAGATION_NOT_SUPPORTED:以非事務方式執行,如果當前存在事務,則把當前事務掛起。
- TransactionDefinition.PROPAGATION_NEVER:以非事務方式執行,如果當前存在事務,則丟擲異常。
- TransactionDefinition.PROPAGATION_MANDATORY:如果當前存在事務,則加入該事務;如果當前沒有事務,則丟擲異常。
- TransactionDefinition.PROPAGATION_NESTED:如果當前存在事務,則建立一個事務作為當前事務的巢狀事務來執行;如果當前沒有事務,則該取值等價於TransactionDefinition.PROPAGATION_REQUIRED。
屬性 | 型別 | 描述 |
---|---|---|
value | String | 可選的限定描述符,指定使用的事務管理器 |
propagation | enum: Propagation | 可選的事務傳播行為設定 |
isolation | enum: Isolation | 可選的事務隔離級別設定 |
readOnly | boolean | 讀寫或只讀事務,預設讀寫 |
timeout | int (in seconds granularity) | 事務超時時間設定 |
rollbackFor | Class物件陣列,必須繼承自Throwable | 導致事務回滾的異常類陣列 |
rollbackForClassName | 類名陣列,必須繼承自Throwable | 導致事務回滾的異常類名字陣列 |
noRollbackFor | Class物件陣列,必須繼承自Throwable | 不會導致事務回滾的異常類陣列 |
noRollbackForClassName | 類名陣列,必須繼承自Throwable | 不會導致事務回滾的異常類名字陣列 |
整合
啟動Spring
web.xml檔案中配置建立Spring容器,讓Spring容器隨著Web應用的啟動而自動啟動,藉助ServletContextListener監聽器完成。實現ContextLoaderListener介面
<listener>
<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>
載入Spring的配置檔案
<!--spring配置檔案的位置-->
<context-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:applicationContext.xml</param-value>
</context-param>
在applicationContext.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:tx="http://www.springframework.org/schema/tx"
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/context
http://www.springframework.org/schema/context/spring-context.xsd
http://www.springframework.org/schema/aop
http://www.springframework.org/schema/aop/spring-aop.xsd
http://www.springframework.org/schema/tx
http://www.springframework.org/schema/tx/spring-tx.xsd">
<!-- 配置自動掃描的包 -->
<context:component-scan base-package="com.qtu404">
<!-- 掃描時跳過 @Controller 註解的JAVA類(控制器) -->
<context:exclude-filter type="annotation"
expression="org.springframework.stereotype.Controller"/>
</context:component-scan>
<!-- 配置db.properyies檔案位置 -->
<bean id="configurer" class="org.springframework.beans.factory.config.PreferencesPlaceholderConfigurer">
<property name="locations" value="classpath:db.properties"></property>
</bean>
<!-- 配置資料來源訊息 -->
<bean id="dataSource" class="org.apache.commons.dbcp.BasicDataSource">
<property name="driverClassName" value="${driver}"/>
<property name="url" value="${url}"></property>
<property name="username" value="${username}"></property>
<property name="password" value="${password}"></property>
</bean>
<!-- 配置SqlSessionFactory -->
<bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean">
<property name="dataSource" ref="dataSource"></property>
<property name="mapperLocations" value="classpath:com/qtu404/mapper/*.xml"></property>
</bean>
<!-- 配置事務管理器 -->
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="dataSource"></property>
</bean>
<!-- 配置事務的傳播性 -->
<tx:advice id="txAdvice" transaction-manager="transactionManager">
<tx:attributes>
<tx:method name="get*" read-only="true"/>
<tx:method name="add*" propagation="REQUIRED"/>
<tx:method name="delete*" propagation="REQUIRED"/>
<tx:method name="update*" propagation="REQUIRED"/>
</tx:attributes>
</tx:advice>
<!-- 配置那些類的哪些方法需要參與事務 -->
<aop:config>
<aop:pointcut
expression="execution(* com.qtu404.user.service.*.*(..))||execution(* com.qtu404.slide.service.*.*(..))"
id="allMethod"/>
<aop:advisor advice-ref="txAdvice" pointcut-ref="allMethod"/>
</aop:config>
</beans>
讓Spring管理控制器
SpringMVC和Spring整合的配置檔案springmvc-servlet.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:mvc="http://www.springframework.org/schema/mvc" xmlns:p="http://www.springframework.org/schema/p"
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/context http://www.springframework.org/schema/context/spring-context-4.1.xsd
http://www.springframework.org/schema/mvc http://www.springframework.org/schema/mvc/spring-mvc-4.1.xsd http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop.xsd">
<!-- scan the package and the sub package -->
<context:component-scan base-package="com.qtu404"/>
<!-- don't handle the static resource -->
<mvc:default-servlet-handler/>
<!-- if you use annotation you must configure following setting -->
<mvc:annotation-driven/>
<!--檔案上傳-->
<bean id="multipartResolver" class="org.springframework.web.multipart.commons.CommonsMultipartResolver"
p:defaultEncoding="UTF-8">
</bean>
<bean id="logger" class="com.qtu404.common.aop.Logger"/>
<aop:config>
<!--使用者操作日誌-->
<aop:aspect id="userLog" ref="logger">
<aop:pointcut id="userPointcut"
expression="execution(* com.qtu404.user.controller.UserController.*(..))"/>
<aop:after method="saveLog" pointcut-ref="userPointcut"/>
</aop:aspect>
<!--幻燈片操作日誌-->
<aop:aspect id="slideLog" ref="logger">
<aop:pointcut id="slidePointcut"
expression="execution(* com.qtu404.slide.controller.SlideController.*(..))"/>
<aop:after method="saveLog" pointcut-ref="slidePointcut"/>
</aop:aspect>
<!--檔案操作日誌-->
<aop:aspect id="fileLog" ref="logger">
<aop:pointcut id="filePointcut"
expression="execution(* com.qtu404.slide.controller.FileController.*(..))"/>
<aop:after method="saveLog" pointcut-ref="filePointcut"/>
</aop:aspect>
<!--資料夾作日誌-->
<aop:aspect id="folderLog" ref="logger">
<aop:pointcut id="folderPointcut"
expression="execution(* com.qtu404.folder.controller.FolderController.*(..))"/>
<aop:after method="saveLog" pointcut-ref="folderPointcut"/>
</aop:aspect>
</aop:config>
<!-- configure the InternalResourceViewResolver -->
<bean class="org.springframework.web.servlet.view.InternalResourceViewResolver"
id="internalResourceViewResolver">
<!-- 字首 -->
<property name="prefix" value="/"/>
<!-- 字尾 -->
<property name="suffix" value=".html"/>
</bean>
</beans>