Tips:若是想要快速查閱的朋友,能夠直接跳轉到 初識AOP(Spring 程序)這一大節php
前面的內容以聯繫過去 Java、JavaWeb 的知識逐步引入到 AOP 爲主 真正的Spring AOP內容包括下面幾個板塊 能夠跳轉一下哈java
(一) AOP 術語mysql
(二) AOP 入門案例:XML 、註解方式spring
(三) 徹底基於 Spring 的事務控制:XML、註解方式、純註解方式sql
在軟件業,AOP爲Aspect Oriented Programming的縮寫,意爲:面向切面編程,經過預編譯方式和運行期間動態代理實現程序功能的統一維護的一種技術。AOP是OOP的延續,是軟件開發中的一個熱點,也是Spring框架中的一個重要內容,是函數式編程的一種衍生範型。利用AOP能夠對業務邏輯的各個部分進行隔離,從而使得業務邏輯各部分之間的耦合度下降,提升程序的可重用性,同時提升了開發的效率。數據庫
—— 百度百科express
開篇就直接來看 Spring AOP 中的百科說明,我我的認爲是很是晦澀的,當回過來頭再看這段引言的時候,才恍然大悟,這段話的意思呢,說白了,就是說咱們把程序中一些重複的代碼拿出來,在須要它執行的時候,能夠經過預編譯或者運行期的動態代理實現不動源碼而動態的給程序進行加強或者添加功能的技術apache
拿出一些重複的代碼? 拿出的到底是什麼代碼呢?舉個例子!編程
在下面的方法中,咱們模擬的是程序中對事務的管理,下面代碼中的 A B均可以看作 「開啓事務」、「提交事務」 的一些事務場景,這些代碼就能夠看作是上面所說的重複的代碼的一種設計模式
而還有一些重複代碼大可能是關於權限管理或者說日誌登陸等一些雖然影響了咱們 代碼業務邏輯的 「乾淨」,可是卻不得不存在,若是有什麼辦法可以抽取出這些方法,使得咱們的業務代碼更加簡潔,天然咱們能夠更專一與咱們的業務,利於開發,這也就是咱們今天想要說重點
最後不得不提的是,AOP 做爲 Spring 這個框架的核心內容之一,很顯然應用了大量的設計模式,設計模式,歸根結底,就是爲了解耦,以及提升靈活性,可擴展性,而咱們所學的一些框架,直接把這些東西封裝好,讓你直接用,說的白一點,就是爲了讓你偷懶,讓你既保持了良好的代碼結構,又不須要和你去本身編寫這些複雜的數據結構,提升了開發效率
一上來就直接談 AOP術語阿,面向切面等等,很顯然不是很合適,光聽名字老是能能讓人 「望文生怯」 , 任何技術的名字只不過是一個名詞罷了,實際上對於入門來講,咱們更須要搞懂的是,經過傳統的程序與使用 Spring AOP 相關技術的程序進行比較,使用 AOP 能夠幫助咱們解決哪些問題或者需求,經過知其然,而後應用其因此然,這樣相比較於,直接學習其基本使用方式,會有靈魂的多!
說明:下面的第一部分的例子是在上一篇文章的程序加以改進,爲了照顧到全部的朋友,我把從依賴到類的編寫都會提到,方便你們有須要來練習,看一下程序的總體結構,對後面的說明也有必定的幫助
說明:因爲我這裏建立的是一個Maven項目,因此在這裏修改 pom.xml 添加一些必要的依賴座標就能夠
若是建立時沒有使用依賴的朋友,去下載咱們所須要的 jar 包導入就能夠了
<packaging>jar</packaging>
<dependencies>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-context</artifactId>
<version>5.0.2.RELEASE</version>
</dependency>
<dependency>
<groupId>mysql</groupId>
<artifactId>mysql-connector-java</artifactId>
<version>5.1.6</version>
</dependency>
<dependency>
<groupId>c3p0</groupId>
<artifactId>c3p0</artifactId>
<version>0.9.1.2</version>
</dependency>
<dependency>
<groupId>commons-dbutils</groupId>
<artifactId>commons-dbutils</artifactId>
<version>1.4</version>
</dependency>
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<version>4.12</version>
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-test</artifactId>
<version>5.0.2.RELEASE</version>
</dependency>
</dependencies>
複製代碼
簡單看一下,spring核心的一些依賴,以及數據庫相關的依賴,還有單元測試等依賴就都導入進來了
下面所要使用的第一個案例,涉及到兩個帳戶之間的模擬轉帳交易,因此咱們建立出含有名稱以及餘額這樣幾個字段的表
-- ----------------------------
-- Table structure for account
-- ----------------------------
CREATE TABLE `account` (
`id` int(11) NOT NULL AUTO_INCREMENT,
`name` varchar(32),
`balance` float,
PRIMARY KEY (`id`)
)
複製代碼
沒什麼好說的,對應着咱們的表創出實體
public class Account implements Serializable {
private Integer id;
private String name;
private Float balance;
......補充 get set toString 方法
複製代碼
下面咱們演示事務問題,最主要仍是使用 transfer 這個轉帳方法,固然還有一些增刪改查的方法,我只留了一個查詢全部的方法,到時候就能夠看出傳統方法中一些代碼的重複以及複雜的工做度
public interface AccountService {
/** * 查詢全部 * @return */
List<Account> findAll();
/** * 轉帳方法 * @param sourceName 轉出帳戶 * @param targetName 轉入帳戶 * @param money */
void transfer(String sourceName,String targetName,Float money);
}
複製代碼
@Service("accountService")
public class AccountServiceImpl implements AccountService {
@Autowired
private AccountDao accountDao;
public List<Account> findAll() {
return accountDao.findAllAccount();
}
public void transfer(String sourceName, String targetName, Float money) {
//根據名稱分別查詢到轉入轉出的帳戶
Account source = accountDao.findAccountByName(sourceName);
Account target = accountDao.findAccountByName(targetName);
//轉入轉出帳戶加減
source.setBalance(source.getBalance() - money);
target.setBalance(target.getBalance() + money);
//更新轉入轉出帳戶
accountDao.updateAccount(source);
accountDao.updateAccount(target);
}
}
複製代碼
public interface AccountDao {
/** * 更細帳戶信息(修改) * @param account */
void updateAccount(Account account);
/** * 查詢全部帳戶 * @return */
List<Account> findAllAccount();
/** * 經過名稱查詢 * @param accountName * @return */
Account findAccountByName(String accountName);
}
複製代碼
咱們引入了 DBUtils 這樣一個操做數據庫的工具,它的做用就是封裝代碼,達到簡化 JDBC 操做的目的,因爲之後整合 SSM 框架的時候,持久層的事情就能夠交給 MyBatis 來作,而今天咱們重點仍是講解 Spring 中的知識,因此這部分會用就能夠了
用到的內容基本講解:
QueryRunner 提供對 sql 語句進行操做的 API (insert delete update)
ResultSetHander 接口,定義了查詢後,如何封裝結果集(僅提供了咱們用到的)
@Repository("accountDao")
public class AccountDaoImpl implements AccountDao {
@Autowired
private QueryRunner runner;
public void updateAccount(Account account) {
try {
runner.update("update account set name=?,balance=? where id=?", account.getName(), account.getBalance(), account.getId());
} catch (Exception e) {
throw new RuntimeException(e);
}
}
public List<Account> findAllAccount() {
try {
return runner.query("select * from account", new BeanListHandler<Account>(Account.class));
} catch (Exception e) {
throw new RuntimeException(e);
}
}
public Account findAccountByName(String accountName) {
try {
List<Account> accounts = runner.query("select * from account where name = ?", new BeanListHandler<Account>(Account.class), accountName);
if (accounts == null || accounts.size() == 0) {
return null;
}
if (accounts.size() > 1) {
throw new RuntimeException("結果集不惟一,數據存在問題");
}
return accounts.get(0);
} catch (Exception e) {
throw new RuntimeException(e);
}
}
}
複製代碼
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans" xmlns:context="http://www.springframework.org/schema/context" 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.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd">
<!--開啓掃描-->
<context:component-scan base-package="cn.ideal"></context:component-scan>
<!--配置 QueryRunner-->
<bean id="runner" class="org.apache.commons.dbutils.QueryRunner">
<!--注入數據源-->
<constructor-arg name="ds" ref="dataSource"></constructor-arg>
</bean>
<!--配置數據源-->
<bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
<property name="driverClass" value="com.mysql.jdbc.Driver"></property>
<property name="jdbcUrl" value="jdbc:mysql://localhost:3306/ideal_spring"></property>
<property name="user" value="root"></property>
<property name="password" value="root99"></property>
</bean>
</beans>
複製代碼
jdbc.driver=com.mysql.jdbc.Driver
jdbc.url=jdbc:mysql://localhost:3306/ideal_spring
jdbc.username=root
jdbc.password=root99
複製代碼
在這裏,咱們使用 Spring以及Junit 測試
說明:使用 @RunWith 註解替換原有運行器 而後使用 @ContextConfiguration 指定 spring 配置文件的位置,而後使用 @Autowired 給變量注入數據
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(locations = "classpath:bean.xml")
public class AccountServiceTest {
@Autowired
private AccountService as;
@Test
public void testFindAll() {
List<Account> list = as.findAll();
for (Account account : list) {
System.out.println(account);
}
}
@Test
public void testTransfer() {
as.transfer("李四", "張三", 500f);
}
}
複製代碼
先執行查詢全部:
再執行模擬轉帳方法:
方法中也就是李四向張三轉帳500,看到下面的結果,是沒有任何問題的
首先分析一下,咱們並無顯式的進行事務的管理,可是不用否認,事務必定存在的,若是沒有提交事務,很顯然,查詢功能是不可以測試成功的,咱們的代碼事務隱式的被自動控制了,使用了 connection 對象的 setAutoCommit(true),即自動提交了
接着看一下配置文件中,咱們只注入了了數據源,這樣作表明什麼呢?
<!--配置 QueryRunner-->
<bean id="runner" class="org.apache.commons.dbutils.QueryRunner">
<!--注入數據源-->
<constructor-arg name="ds" ref="dataSource"></constructor-arg>
</bean>
複製代碼
也就是說,每一條語句獨立事務:
說白了,就是各管各的,彼此沒任何溝通,例如在Service的轉帳方法中,下面標着 1 2 3 4 5 的位置處的語句,每個調用時,都會建立一個新的 QueryRunner對象,而且從數據源中獲取一個鏈接,可是,當在某一個步驟中忽然出現問題,前面的語句仍然會執行,可是後面的語句就由於異常而終止了,這也就是咱們開頭說的,彼此之間是獨立的
public void transfer(String sourceName, String targetName, Float money) {
//根據名稱分別查詢到轉入轉出的帳戶
Account source = accountDao.findAccountByName(sourceName); // 1
Account target = accountDao.findAccountByName(targetName); // 2
//轉入轉出帳戶加減
source.setBalance(source.getBalance() - money); // 3
target.setBalance(target.getBalance() + money);
//更新轉入轉出帳戶
accountDao.updateAccount(source); // 4
//模擬轉帳異常
int num = 100/0; // 異常
accountDao.updateAccount(target); //5
}
複製代碼
很顯然這是很是不合適的,甚至是致命的,像咱們代碼中所寫,轉出帳戶的帳戶信息已經扣款更新了,可是轉入方的帳戶信息卻因爲前面異常的發生,致使並無成功執行,李四從2500 變成了 2000,可是張三卻沒有成功收到轉帳
上面出現的問題,歸根結底是因爲咱們持久層中的方法獨立事務,因此沒法實現總體的事務控制(與事務的一致性相悖)那麼咱們解決問題的思路是什麼呢?
首先咱們須要作的,就是使用 ThreadLocal 對象把 Connection 和當前線程綁定,從而使得一個線程中只有一個控制事務的對象,
簡單提一下Threadlocal:
Threadlocal 是一個線程內部的存儲類,能夠在指定線程內存儲數據,也就至關於,這些數據就被綁定在這個線程上了,只能經過這個指定的線程,才能夠獲取到想要的數據
這是官方的說明:
This class provides thread-local variables. These variables differ from their normal counterparts in that each thread that accesses one (via its get or set method) has its own, independently initialized copLy of the variable. ThreadLocal instances are typically private static fields in classes that wish to associate state with a thread (e.g., a user ID or Transaction ID).
就是說,ThreadLoacl 提供了線程內存儲局部變量的方式,這些變量比較特殊的就是,每個線程獲取到的變量都是獨立的,獲取數據值的方法就是 get 以及 set
建立 utils 包 ,而後建立一個 ConnectionUtils 工具類,其中最主要的部分,其實也就是寫了一個簡單的判斷,若是這個線程中已經存在鏈接,就直接返回,若是不存在鏈接,就獲取數據源中的一個連接,而後存入,再返回
@Component
public class ConnectionUtils {
private ThreadLocal<Connection> threadLocal = new ThreadLocal<Connection>();
@Autowired
private DataSource dataSource;
public Connection getThreadConnection() {
try {
// 從 ThreadLocal獲取
Connection connection = threadLocal.get();
//先判斷是否爲空
if (connection == null) {
//從數據源中獲取一個鏈接,且存入 ThreadLocal
connection = dataSource.getConnection();
threadLocal.set(connection);
}
return connection;
} catch (Exception e) {
throw new RuntimeException(e);
}
}
public void removeConnection(){
threadLocal.remove();
}
}
複製代碼
接着能夠建立一個管理事務的工具類,其中包括,開啓、提交、回滾事務,以及釋放鏈接
@Component
public class TransactionManager {
@Autowired
private ConnectionUtils connectionUtils;
/** * 開啓事務 */
public void beginTransaction() {
try {
connectionUtils.getThreadConnection().setAutoCommit(false);
} catch (Exception e) {
e.printStackTrace();
}
}
/** * 提交事務 */
public void commit() {
try {
connectionUtils.getThreadConnection().commit();
} catch (Exception e) {
e.printStackTrace();
}
}
/** * 回滾事務 */
public void rollback() {
try {
System.out.println("回滾事務" + connectionUtils.getThreadConnection());
connectionUtils.getThreadConnection().rollback();
} catch (Exception e) {
e.printStackTrace();
}
}
/** * 釋放鏈接 */
public void release() {
try {
connectionUtils.getThreadConnection().close();//還回鏈接池中
connectionUtils.removeConnection();
} catch (Exception e) {
e.printStackTrace();
}
}
}
複製代碼
在方法中添加事務管理的代碼,正常狀況下執行開啓事務,執行操做(你的業務代碼),提交事務,捕獲到異常後執行回滾事務操做,最終執行釋放鏈接
在這種狀況下,即便在某個步驟中出現了異常狀況,也不會對數據形成實際的更改,這樣上面的問題就初步解決了
@Service("accountService")
public class AccountServiceImpl implements AccountService {
@Autowired
private AccountDao accountDao;
@Autowired
private TransactionManager transactionManager;
public List<Account> findAll() {
try {
//開啓事務
transactionManager.beginTransaction();
//執行操做
List<Account> accounts = accountDao.findAllAccount();
//提交事務
transactionManager.commit();
//返回結果
return accounts;
} catch (Exception e) {
//回滾操做
transactionManager.rollback();
throw new RuntimeException(e);
} finally {
//釋放鏈接
transactionManager.release();
}
}
public void transfer(String sourceName, String targetName, Float money) {
try {
//開啓事務
transactionManager.beginTransaction();
//執行操做
//根據名稱分別查詢到轉入轉出的帳戶
Account source = accountDao.findAccountByName(sourceName);
Account target = accountDao.findAccountByName(targetName);
//轉入轉出帳戶加減
source.setBalance(source.getBalance() - money);
target.setBalance(target.getBalance() + money);
//更新轉出轉入帳戶
accountDao.updateAccount(source);
//模擬轉帳異常
int num = 100 / 0;
accountDao.updateAccount(target);
//提交事務
transactionManager.commit();
} catch (Exception e) {
//回滾操做
transactionManager.rollback();
e.printStackTrace();
} finally {
//釋放鏈接
transactionManager.release();
}
}
}
複製代碼
雖然上面,咱們已經實現了在業務層進行對事務的控制,可是很顯然能夠看見,咱們在每個方法中都存在着太多重複的代碼了,而且以業務層與事務管理的方法出現了耦合,打個比方,事務管理類中的隨便一個方法名進行更改,就會直接致使業務層中找不到對應的方法,所有須要修改,若是在業務層方法較多時,很顯然這是很麻煩的
這種狀況下,咱們能夠經過使用靜態代理這一種方式,來進行對上面程序的改進,改進以前爲了照顧到全部的朋友,回顧一下動態代理的一個介紹以及基本使用方式
動態代理,也就是給某個對象提供一個代理對象,用來控制對這個對象的訪問
簡單的舉個例子就是:買火車、飛機票等,咱們能夠直接從車站售票窗口進行購買,這就是用戶直接在官方購買,可是咱們不少地方的店鋪或者一些路邊的亭臺中均可以進行火車票的代售,用戶直接能夠在代售點購票,這些地方就是代理對象
這個動態代理的優點,帶給咱們不少方便,它能夠幫助咱們實現無侵入式的代碼擴展,也就是在不用修改源碼的基礎上,同時加強方法
動態代理分爲兩種:① 基於接口的動態代理 ② 基於子類的動態代理
A:建立官方售票處(類和接口)
RailwayTicketProducer 接口
/** * 生產廠家的接口 */
public interface RailwayTicketProducer {
public void saleTicket(float price);
public void ticketService(float price);
}
複製代碼
RailwayTicketProducerImpl 類
實現類中,咱們後面只對銷售車票方法進行了加強,售後服務並無涉及到
/** * 生產廠家具體實現 */
public class RailwayTicketProducerImpl implements RailwayTicketProducer{
public void saleTicket(float price) {
System.out.println("銷售火車票,收到車票錢:" + price);
}
public void ticketService(float price) {
System.out.println("售後服務(改簽),收到手續費:" + price);
}
}
複製代碼
Client 類
這個類,就是客戶類,在其中,經過代理對象,實現購票的需求
首先先來講一下如何建立一個代理對象:答案是 Proxy類中的 newProxyInstance 方法
注意:既然叫作基於接口的動態代理,這就是說被代理的類,也就是文中官方銷售車票的類最少必須實現一個接口,這是必要的!
public class Client {
public static void main(String[] args) {
RailwayTicketProducer producer = new RailwayTicketProducerImpl();
//動態代理
RailwayTicketProducer proxyProduce = (RailwayTicketProducer) Proxy.newProxyInstance(producer.getClass().getClassLoader(),
producer.getClass().getInterfaces(),new MyInvocationHandler(producer));
//客戶經過代理買票
proxyProduce.saleTicket(1000f);
}
}
複製代碼
newProxyInstance 共有三個參數 來解釋一下:
ClassLoader:類加載器
Class[]:字節碼數組
InvocationHandler:如何代理,也就是想要加強的方式
也就是說,咱們主須要 new 出 InvocationHandler,而後書寫其實現類,是否寫成匿名內部類能夠本身選擇
如上述代碼中 new MyInvocationHandler(producer) 實例化的是我本身編寫的一個 MyInvocationHandler類,實際上能夠在那裏直接 new 出 InvocationHandler,而後重寫其方法,其本質也是經過實現 InvocationHandler 的 invoke 方法實現加強
MyInvocationHandler 類
這個 invoke 方法具備攔截的功能,被代理對象的任何方法被執行,都會通過 invoke
public class MyInvocationHandler implements InvocationHandler {
private Object implObject ;
public MyInvocationHandler (Object implObject){
this.implObject=implObject;
}
/** * 做用:執行被代理對象的任何接口方法都會通過該方法 * 方法參數的含義 * @param proxy 代理對象的引用 * @param method 當前執行的方法 * @param args 當前執行方法所需的參數 * @return 和被代理對象方法有相同的返回值 * @throws Throwable */
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
Object returnValue = null;
//獲取方法執行的參數
Float price = (Float)args[0];
//判斷是否是指定方法(以售票爲例)
if ("saleTicket".equals(method.getName())){
returnValue = method.invoke(implObject,price*0.8f);
}
return returnValue;
}
}
複製代碼
在此處,咱們獲取到客戶購票的金額,因爲咱們使用了代理方進行購票,因此代理方會收取必定的手續費,因此用戶提交了 1000 元,實際上官方收到的只有800元,這也就是這種代理的實現方式,結果以下
銷售火車票,收到車票錢:800.0
上面方法簡單的實現起來也不是很難,可是惟一的標準就是,被代理對象必須提供一個接口,而如今所講解的這一種就是一種能夠直接代理普通 Java 類的方式,同時在演示的時候,我會將代理方法直接之內部類的形式寫出,就不單首創建類了,方便你們與上面對照
增長 cglib 依賴座標
<dependencies>
<dependency>
<groupId>cglib</groupId>
<artifactId>cglib</artifactId>
<version>3.2.4</version>
</dependency>
</dependencies>
複製代碼
TicketProducer 類
/** * 生產廠家 */
public class TicketProducer {
public void saleTicket(float price) {
System.out.println("銷售火車票,收到車票錢:" + price);
}
public void ticketService(float price) {
System.out.println("售後服務(改簽),收到手續費:" + price);
}
}
複製代碼
Enhancer 類中的 create 方法就是用來建立代理對象的
而 create 方法又有兩個參數
public class Client {
public static void main(String[] args) {
// 因爲下方匿名內部類,須要在此處用final修飾
final TicketProducer ticketProducer = new TicketProducer();
TicketProducer cglibProducer =(TicketProducer) Enhancer.create(ticketProducer.getClass(), new MethodInterceptor() {
/** * 前三個三個參數和基於接口的動態代理中invoke方法的參數是同樣的 * @param o * @param method * @param objects * @param methodProxy 當前執行方法的代理對象 * @return * @throws Throwable */
public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
Object returnValue = null;
//獲取方法執行的參數
Float price = (Float)objects[0];
//判斷是否是指定方法(以售票爲例)
if ("saleTicket".equals(method.getName())){
returnValue = method.invoke(ticketProducer,price*0.8f);
}
return returnValue;
}
});
cglibProducer.saleTicket(900f);
}
複製代碼
在這裏咱們寫一個用於建立業務層對象的工廠
在這段代碼中,咱們使用了前面所回顧的基於接口的動態代理方式,在執行方法的先後,分別寫入了開啓事務,提交事務,回滾事務等事務管理方法,這時候,業務層就能夠刪除掉前面所寫的關於業務的重複代碼
@Component
public class BeanFactory {
@Autowired
private AccountService accountService;
@Autowired
private TransactionManager transactionManager;
@Bean("proxyAccountService")
public AccountService getAccountService() {
return (AccountService) Proxy.newProxyInstance(accountService.getClass().getClassLoader(),
accountService.getClass().getInterfaces(),
new InvocationHandler() {
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
Object returnValue = null;
try {
//開啓事務
transactionManager.beginTransaction();
//執行操做
returnValue = method.invoke(accountService, args);
//提交事務
transactionManager.commit();
//返回結果
return returnValue;
} catch (Exception e) {
//回滾事務
transactionManager.rollback();
throw new RuntimeException();
} finally {
//釋放鏈接
transactionManager.release();
}
}
});
}
}
複製代碼
AccountServiceTest 類
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(locations = "classpath:bean.xml")
public class AccountServiceTest {
@Autowired
@Qualifier("proxyAccountService")
private AccountService as;
@Test
public void testFindAll() {
List<Account> list = as.findAll();
for (Account account : list) {
System.out.println(account);
}
}
@Test
public void testTransfer() {
as.transfer("李四", "張三", 500f);
}
}
複製代碼
到如今,一個相對完善的案例就改造完成了,因爲咱們上面大致使用的是註解的方式,並無所有使用 XML 進行配置,若是使用 XML 進行配置,配置也是相對繁瑣的,那麼咱們鋪墊這麼多的內容,實際上就是爲了引出 Spring 中 AOP 的概念,從根源上,一步一步,根據問題引出要學習的技術
讓咱們一塊兒來看一看!
在前面,大篇幅的講解咱們在傳統的程序中,是如何一步一步,改進以及處理例如事務這樣的問題的,而 Spring 中 AOP 這個技術,就能夠幫助咱們來在不修源碼的基礎上對已經存在的方法進行加強,一樣維護也是很方便,大大的提升了開發的效率,如今咱們開始正式介紹 AOP 的知識,有了必定的知識鋪墊後,就可使用 AOP 的方式繼續對前面的程序進行改進!
任何一門技術,都會有其特定的術語,實際上就是一些特定的名稱而已,事實上,我之前在學習的時候,感受 AOP 的一些術語都是相對抽象的,並無很直觀的體現出它的意義,可是這些術語已經普遍的被開發者熟知,成爲了在這個相關技術中,默認已知的一些概念,雖然更重要的是理解 AOP 的思想與使用方式,可是,咱們仍是須要講這樣一種 「共識」 介紹一下
《Spring 實戰》中有這樣一句話,摘出來:
在咱們進入某個領域以前,必須學會在這個領域該如何說話
通知(Advice)
鏈接點(Joinpoint)
切入點(Pointcut)
切面(Aspect)
引入(Introduction)
織入(Weaving)
首先,經過一個很是簡單的案例,來演示一下,如何在某幾個方法執行前,均執行一個日誌的打印方法,簡單模擬爲輸出一句話,前面的步驟咱們都很熟悉,須要注意的就是 bean.xml 中配置的方法,我會代碼下面進行詳的講解
aspectjweaver,這個依賴用來支持切入點表達式等,後面配置中會提到這個知識
<packaging>jar</packaging>
<dependencies>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-context</artifactId>
<version>5.0.2.RELEASE</version>
</dependency>
<dependency>
<groupId>org.aspectj</groupId>
<artifactId>aspectjweaver</artifactId>
<version>1.8.7</version>
</dependency>
</dependencies>
複製代碼
AccountService 接口
public interface AccountService {
/** * 保存帳戶 */
void addAccount();
/** * 刪除帳戶 * @return */
int deleteAccount();
/** * 更新帳戶 * @param i */
void updateAccount(int i);
}
複製代碼
AccountServiceImpl 實現類
public class AccountServiceImpl implements AccountService {
public void addAccount() {
System.out.println("這是增長方法");
}
public int deleteAccount() {
System.out.println("這是刪除方法");
return 0;
}
public void updateAccount(int i) {
System.out.println("這是更新方法");
}
}
複製代碼
public class Logger {
/** * 用於打印日誌:計劃讓其在切入點方法執行以前執行(切入點方法就是業務層方法) */
public void printLog(){
System.out.println("Logger類中的printLog方法執行了");
}
}
複製代碼
bean.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: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/aop http://www.springframework.org/schema/aop/spring-aop.xsd">
<!--配置Spring的IOC,配置service進來-->
<bean id="accountService" class="cn.ideal.service.impl.AccountServiceImpl"></bean>
<!--配置 Logger 進來-->
<bean id="logger" class="cn.ideal.utils.Logger"></bean>
<!--配置 AOP-->
<aop:config>
<!--配置切面-->
<aop:aspect id="logAdvice" ref="logger">
<!--通知的類型,以及創建通知方法和切入點方法的關聯-->
<aop:before method="printLog" pointcut="execution(* cn.ideal.service.impl.*.*(..))"></aop:before>
</aop:aspect>
</aop:config>
</beans>
複製代碼
<?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: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/aop http://www.springframework.org/schema/aop/spring-aop.xsd">
</beans>
複製代碼
首先須要引入的就是這個XML的頭部文件,一些約束,能夠直接複製這裏的,也能夠像之前同樣,去官網找對應的約束等
接着,將 Service 和 Logger 經過 bean 標籤配置進來
aop:config:代表開始 aop 配置,配置的代碼所有寫在這個標籤內
aop:aspect:代表開始配置切面
aop:aspect 標籤內部,經過對應的標籤,配置通知的類型
<aop:config>
<!--配置切面-->
<aop:aspect id="logAdvice" ref="logger">
<!--通知的類型,以及創建通知方法和切入點方法的關聯-->
</aop:aspect>
</aop:config>
複製代碼
題目中咱們是以在方法執行前執行通知,因此是使用了前置通知
aop:before:用於配置前置通知,指定加強的方法在切入點方法以前執行
aop:after-returning:用於配置後置通知,與異常通知只能執行其中一個
aop:after-throwing:用於配置異常通知,異常通知只能執行其中一個
aop:after:用於配置最終通知,不管切入點方法執行時是否有異常,它都會在其後面執行
參數:
method:用於指定通知類中的加強方法名稱,也就是咱們上面的 Logger類中的 printLog 方法
poinitcut:用於指定切入點表達式(文中使用的是這個)指的是對業務層中哪些方法進行加強
ponitcut-ref:用於指定切入點的表達式的引用(調用次數過多時,更多的使用這個,減小了重複的代碼)
切入點表達式的寫法:
首先,在poinitcut屬性的引號內 加入execution() 關鍵字,括號內書寫表達式
基本格式:訪問修飾符 返回值 包名.包名.包名...類名.方法名(方法參數)
說明:包名有幾個是根據本身的類全部在的包結構決定
全匹配寫法
public void cn.ideal.service.impl.AccountServiceImpl.addAccount()
訪問修飾符,如 public 能夠省略,返回值可使用通配符,表示任意返回值
void cn.ideal.service.impl.AccountServiceImpl.addAccount()
包名可使用通配符,表示任意包,有幾級包,就須要寫幾個*.
* *.*.*.*.AccountServiceImpl.addAccount()
包名可使用..表示當前包及其子包
cn..*.addAccount()
類名和方法名均可以使用*來實現通配,下面表示全通配
* *..*.*(..)
方法參數
能夠直接寫數據類型:例如 int
引用類型寫包名.類名的方式 java.lang.String
可使用通配符表示任意類型,可是必須有參數
可使用..表示有無參數都可,有參數能夠是任意類型
在實際使用中,更加推薦的寫法也就是上面代碼中的那種,將包結構給出(通常都是對業務層加強),其餘的使用通配符
pointcut="execution(* cn.ideal.service.impl.*.*(..))"
在給出4中通知類型後,就須要屢次書寫這個切入表達式,因此咱們可使用 pointcut-ref 參數解決重複代碼的問題,其實就至關於抽象出來了,方便之後調用
ponitcut-ref:用於指定切入點的表達式的引用(調用次數過多時,更多的使用這個,減小了重複的代碼)
位置放在 config裏,aspect 外就能夠了
<aop:pointcut id="pt1" expression="execution(* cn.ideal.service.impl.*.*(..))"></aop:pointcut>
複製代碼
調用時:
<aop:before method="PrintLog" pointcut-ref="pt1"></aop:before>
複製代碼
接着,spring框架爲咱們提供的一種能夠手動在代碼中控制加強代碼何時執行的方式,也就是環繞通知
配置中須要這樣一句話,pt1和前面是同樣的
<aop:around method="aroundPrintLog" pointcut-ref="pt1"></aop:around>
複製代碼
Logger類中這樣配置
public Object aroundPrintLog(ProceedingJoinPoint proceedingJoinPoint) {
Object returValue = null;
try {
Object[] args = proceedingJoinPoint.getArgs();
System.out.println("這是Logger類中的aroundPrintLog前置方法");
returValue = proceedingJoinPoint.proceed(args);
System.out.println("這是Logger類中的aroundPrintLog後置方法");
return returValue;
} catch (Throwable throwable) {
System.out.println("這是Logger類中的aroundPrintLog異常方法");
throw new RuntimeException();
} finally {
System.out.println("這是Logger類中的aroundPrintLog最終方法");
}
}
複製代碼
來解釋一下:
Spring 中提供了一個接口:ProceedingJoinPoint,其中有一個方法叫作 proceed(args),這個方法就至關於明確調用切入點方法,proceed() 方法就好像之前動態代理中的 invoke,同時這個接口能夠做爲環繞通知的方法參數,這樣看起來,和前面的動態代理的那種感受仍是很類似的
依賴,以及業務層方法,咱們都是用和 XML 一致的嗎,不過爲了演示方便,這裏就只留下 一個 add 方法
配置文件中一個是須要引入新的約束,再有就是開啓掃描以及開啓註解 AOP 的支持
<?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:aop="http://www.springframework.org/schema/aop" xmlns:context="http://www.springframework.org/schema/context" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd">
<!-- 配置spring建立容器時要掃描的包-->
<context:component-scan base-package="cn.ideal"></context:component-scan>
<!-- 配置spring開啓註解AOP的支持 -->
<aop:aspectj-autoproxy></aop:aspectj-autoproxy>
</beans>
複製代碼
首先是業務層中把 Service 注進來
@Service("accountService")
public class AccountServiceImpl implements AccountService {
public void addAccount() {
System.out.println("這是增長方法");
}
}
複製代碼
接着就是最終要的位置Logger類中,首先將這個類經過 @Component("logger") 總體注入
而後使用 @Aspect 代表這是一個切面類
下面我分別使用了四種通知類型,以及環繞通知類型,在註解中這裏是須要注意的
第一次我首先測試的是四種通知類型:將環繞通知先註釋掉,把前面四個放開註釋
@Component("logger")
@Aspect//表示當前類是一個切面類
public class Logger {
@Pointcut("execution(* cn.ideal.service.impl.*.*(..))")
private void pt1(){}
// @Before("pt1()")
public void printLog1(){
System.out.println("Logger類中的printLog方法執行了-前置");
}
// @AfterReturning("pt1()")
public void printLog2(){
System.out.println("Logger類中的printLog方法執行了-後置");
}
// @AfterThrowing("pt1()")
public void printLog3(){
System.out.println("Logger類中的printLog方法執行了-異常");
}
// @After("pt1()")
public void printLog4(){
System.out.println("Logger類中的printLog方法執行了-最終");
}
@Around("pt1()")
public Object aroundPrintLog(ProceedingJoinPoint proceedingJoinPoint) {
Object returValue = null;
try {
Object[] args = proceedingJoinPoint.getArgs();
System.out.println("這是Logger類中的aroundPrintLog前置方法");
returValue = proceedingJoinPoint.proceed(args);
System.out.println("這是Logger類中的aroundPrintLog後置方法");
return returValue;
} catch (Throwable throwable) {
System.out.println("這是Logger類中的aroundPrintLog異常方法");
throw new RuntimeException();
} finally {
System.out.println("這是Logger類中的aroundPrintLog最終方法");
}
}
}
複製代碼
四種通知類型測試結果:
能夠看到,一個特別詭異的事情出現了,後置通知和最終通知的位置出現了問題,一樣異常狀況下也會出現這樣的問題,確實這是這裏的一個問題,因此咱們註解中通常使用 環繞通知的方式
環繞通知測試結果:
純註解仍是比較簡單的 加好 @EnableAspectJAutoProxy 就能夠了
@Configuration
@ComponentScan(basePackages="cn.ideal")
@EnableAspectJAutoProxy//主要是這個註解
public class SpringConfiguration {
}
複製代碼
到這裏,兩種XML以及註解兩種方式的基本使用就都說完了,下面咱們會講一講如何徹底基於 Spring 實現事務的控制
上面Spring中 AOP 知識的入門,可是實際上,Spring 做爲一個強大的框架,爲咱們業務層中事務處理,已經進行了考慮,它爲咱們提供了一組關於事務控制的接口,基於 AOP 的基礎之上,就能夠高效的完成事務的控制,下面咱們就經過一個案例,來對這部份內容進行介紹,這一部分,咱們選用的的例如 持久層 單元測試等中的內容均使用 Spring,特別注意:持久層咱們使用的是 Spring 的 JdbcTemplate ,不熟悉的朋友能夠去簡單瞭解一下,在這個案例中,重點仍是學習事務的控制,這裏不會形成太大的影響的
注:準備完代碼第一個要演示的是基於 XML 的形式,因此咱們準備的時候都沒有使用註解,後面介紹註解方式的時候,會進行修改
<packaging>jar</packaging>
<dependencies>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-context</artifactId>
<version>5.0.2.RELEASE</version>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-jdbc</artifactId>
<version>5.0.2.RELEASE</version>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-tx</artifactId>
<version>5.0.2.RELEASE</version>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-test</artifactId>
<version>5.0.2.RELEASE</version>
</dependency>
<dependency>
<groupId>mysql</groupId>
<artifactId>mysql-connector-java</artifactId>
<version>5.1.6</version>
</dependency>
<dependency>
<groupId>org.aspectj</groupId>
<artifactId>aspectjweaver</artifactId>
<version>1.8.7</version>
</dependency>
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<version>4.12</version>
</dependency>
</dependencies>
複製代碼
建立 Account 表
-- ----------------------------
-- Table structure for account
-- ----------------------------
CREATE TABLE `account` (
`id` int(11) NOT NULL AUTO_INCREMENT,
`name` varchar(32),
`balance` float,
PRIMARY KEY (`id`)
)
複製代碼
建立 Account 類
沒什麼好說的,對應着咱們的表創出實體
public class Account implements Serializable {
private Integer id;
private String name;
private Float balance;
......補充 get set toString 方法
複製代碼
爲了減小篇幅,就給了實現類,接口就不貼了,這很簡單
業務層
package cn.ideal.service.impl;
import cn.ideal.dao.AccountDao;
import cn.ideal.domain.Account;
import cn.ideal.service.AccountService;
public class AccountServiceImpl implements AccountService {
private AccountDao accountDao;
public void setAccountDao(AccountDao accountDao) {
this.accountDao = accountDao;
}
public Account findAccountById(Integer accountId) {
return accountDao.findAccountById(accountId);
}
public void transfer(String sourceName, String targetName, Float money) {
System.out.println("轉帳方法執行");
//根據名稱分別查詢到轉入轉出的帳戶
Account source = accountDao.findAccountByName(sourceName);
Account target = accountDao.findAccountByName(targetName);
//轉入轉出帳戶加減
source.setBalance(source.getBalance() - money);
target.setBalance(target.getBalance() + money);
//更新轉入轉出帳戶
accountDao.updateAccount(source);
int num = 100/0;
accountDao.updateAccount(target);
}
}
複製代碼
持久層
public class AccountDaoImpl extends JdbcDaoSupport implements AccountDao {
public Account findAccountById(Integer accountId) {
List<Account> accounts = super.getJdbcTemplate().query("select * from account where id = ?",new BeanPropertyRowMapper<Account>(Account.class),accountId);
return accounts.isEmpty()?null:accounts.get(0);
}
public Account findAccountByName(String accountName) {
List<Account> accounts = super.getJdbcTemplate().query("select * from account where name = ?",new BeanPropertyRowMapper<Account>(Account.class),accountName);
if(accounts.isEmpty()){
return null;
}
if(accounts.size()>1){
throw new RuntimeException("結果集不惟一");
}
return accounts.get(0);
}
public void updateAccount(Account account) {
super.getJdbcTemplate().update("update account set name=?,balance=? where id=?",account.getName(),account.getBalance(),account.getId());
}
}
複製代碼
提一句:若是沒有用過 JdbcTemplate,可能會好奇下面的 DriverManagerDataSource 是什麼,這個是 Spring 內置的數據源
<?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.xsd">
<!-- 配置業務層-->
<bean id="accountService" class="cn.ideal.service.impl.AccountServiceImpl">
<property name="accountDao" ref="accountDao"></property>
</bean>
<!-- 配置帳戶的持久層-->
<bean id="accountDao" class="cn.ideal.dao.impl.AccountDaoImpl">
<property name="dataSource" ref="dataSource"></property>
</bean>
<!-- 配置數據源-->
<bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource">
<property name="driverClassName" value="com.mysql.jdbc.Driver"></property>
<property name="url" value="jdbc:mysql://localhost:3306/ideal_spring"></property>
<property name="username" value="root"></property>
<property name="password" value="root99"></property>
</bean>
</beans>
複製代碼
/** * 使用Junit單元測試:測試咱們的配置 */
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(locations = "classpath:bean.xml")
public class AccountServiceTest {
@Autowired
private AccountService as;
@Test
public void testTransfer() {
as.transfer("張三", "李四", 500f);
}
複製代碼
首先要作的就是修改配置文件,這裏須要引入的就是 aop 和 tx 這兩個名稱空間
配置 業務層 持久層 以及數據源 沒什麼好說的,直接複製過來,下面就是咱們真正的重要配置
真正管理事務的對象 Spring 已經提供給咱們了
使用Spring JDBC或iBatis 進行持久化數據時可使用 org.springframework.jdbc.datasource.DataSourceTransactionManager
使用 Hibernate 進行持久化數據時可使用org.springframework.orm.hibernate5.HibernateTransactionManager
在其中將數據源引入
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="dataSource"></property>
</bean>
複製代碼
進行事務通知以及屬性配置時就須要引入事務的約束,tx 以及 aop 的名稱空間和約束
在這裏,就能夠將事務管理器引入
<!-- 配置事務的通知-->
<tx:advice id="txAdvice" transaction-manager="transactionManager">
</tx:advice>
複製代碼
在 <tx:advice></tx:advice>
中就能夠配置事務的屬性了,這裏有一些屬性須要熟悉一下,關於事務的隔離級別能夠暫時看一看就能夠了,只針對這個例程的話,咱們並無太多的涉及,事務是一個大問題,須要深刻的瞭解,咱們在這裏更重點講的是如何配置使用它
name:指定你須要增長某種事務的方法名,可使用通配符,例如 * 表明全部 find* 表明名稱開頭爲 find 的方法,第二種優先級要更高一些
isolation:用於指定事務的隔離級別,表示使用數據庫的默認隔離級別,默認值是DEFAULT
未提交讀取(Read Uncommitted)
Spring標識:ISOLATION_READ_UNCOMMITTED
表明容許髒讀取,但不容許更新丟失。也就是說,若是一個事務已經開始寫數據,則另一個事務則不容許同時進行寫操做,但容許其餘事務讀此行數據
已提交讀取(Read Committed)
Spring標識:ISOLATION_READ_COMMITTED
只能讀取已經提交的數據,解決了髒讀的問題。讀取數據的事務容許其餘事務繼續訪問該行數據,可是未提交的寫事務將會禁止其餘事務訪問該行
可重複讀取(Repeatable Read)
序列化(Serializable)
propagation:用於指定事務的傳播屬性,默認值是 REQUIRED,表明必定會有事務,通常被用於增刪改,查詢方法能夠選擇使用 SUPPORTS
read-only:用於指定事務是否只讀。默認值是false示讀寫,通常查詢方法才設置爲true
timeout:用於指定事務的超時時間,默認值是-1,表示永不超時,若是指定了數值,以秒爲單位,通常不會用這個屬性
rollback-for:用於指定一個異常,當產生該異常時,事務回滾,產生其餘異常時,事務不回滾。沒有默認值。表示任何異常都回滾
no-rollback-for:用於指定一個異常,當產生該異常時,事務不回滾,產生其餘異常時事務回滾。沒有默認值。表示任何異常都回滾
<tx:advice id="txAdvice" transaction-manager="transactionManager">
<!-- 配置事務的屬性 -->
<tx:attributes>
<tx:method name="*" propagation="REQUIRED" read-only="false"/>
<tx:method name="find*" propagation="SUPPORTS" read-only="true"/>
</tx:attributes>
</tx:advice>
複製代碼
<!-- 配置aop-->
<aop:config>
!-- 配置切入點表達式-->
<aop:pointcut id="pt1" expression="execution(* cn.ideal.service.impl.*.*(..))"></aop:pointcut>
</aop:config>
複製代碼
在 <aop:config></aop:config>
中進行此步驟
<!-- 配置aop-->
<aop:config>
!-- 配置切入點表達式-->
<aop:pointcut id="pt1" expression="execution(* cn.ideal.service.impl.*.*(..))"></aop:pointcut>
<!--創建切入點表達式和事務通知的對應關係 -->
<aop:advisor advice-ref="txAdvice" pointcut-ref="pt1"></aop:advisor>
</aop:config>
複製代碼
<?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: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.xsd http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx.xsd http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop.xsd">
<!-- 配置業務層-->
<bean id="accountService" class="cn.ideal.service.impl.AccountServiceImpl">
<property name="accountDao" ref="accountDao"></property>
</bean>
<!-- 配置帳戶的持久層-->
<bean id="accountDao" class="cn.ideal.dao.impl.AccountDaoImpl">
<property name="dataSource" ref="dataSource"></property>
</bean>
<!-- 配置數據源-->
<bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource">
<property name="driverClassName" value="com.mysql.jdbc.Driver"></property>
<property name="url" value="jdbc:mysql://localhost:3306/ideal_spring"></property>
<property name="username" value="root"></property>
<property name="password" value="root99"></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="*" propagation="REQUIRED" read-only="false"/>
<tx:method name="find*" propagation="SUPPORTS" read-only="true"/>
</tx:attributes>
</tx:advice>
<!-- 配置aop-->
<aop:config>
<!-- 配置切入點表達式-->
<aop:pointcut id="pt1" expression="execution(* cn.ideal.service.impl.*.*(..))"></aop:pointcut>
<!--創建切入點表達式和事務通知的對應關係 -->
<aop:advisor advice-ref="txAdvice" pointcut-ref="pt1"></aop:advisor>
</aop:config>
</beans>
複製代碼
仍是基本的代碼,可是須要對持久層進行一個小小的修改,前面爲了配置中簡單一些,咱們直接使用了繼承 JdbcDaoSupport 的方式,可是它只能用於 XML 的方式, 註解是不能夠這樣用的,因此,咱們仍是須要用傳統的一種方式,也就是在 Dao 中定義 JdcbTemplate
註解的常規操做,開啓註解,咱們這裏把數據源和JdbcTemplate也配置好
<?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:aop="http://www.springframework.org/schema/aop" xmlns:tx="http://www.springframework.org/schema/tx" xmlns:context="http://www.springframework.org/schema/context" xsi:schemaLocation=" http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx.xsd http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd">
<!-- 配置spring建立容器時要掃描的包-->
<context:component-scan base-package="cn.ideal"></context:component-scan>
<!-- 配置JdbcTemplate-->
<bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
<property name="dataSource" ref="dataSource"></property>
</bean>
<!-- 配置數據源-->
<bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource">
<property name="driverClassName" value="com.mysql.jdbc.Driver"></property>
<property name="url" value="jdbc:mysql://localhost:3306/ideal_spring"></property>
<property name="username" value="root"></property>
<property name="password" value="root99"></property>
</bean>
</beans>
複製代碼
@Service("accountService")
public class AccountServiceImpl implements AccountService {
@Autowired
private AccountDao accountDao;
//下面是同樣的
}
複製代碼
@Repository("accountDao")
public class AccountDaoImpl implements AccountDao {
@Autowired
private JdbcTemplate jdbcTemplate;
//下面基本是同樣的
//只須要將原來的 super.getJdbcTemplate().xxx 改成直接用 jdbcTemplate 執行
}
複製代碼
<!-- 配置事務管理器 -->
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="dataSource"></property>
</bean>
複製代碼
<!-- 開啓spring對註解事務的支持-->
<tx:annotation-driven transaction-manager="transactionManager"></tx:annotation-driven>
複製代碼
這個註解能夠出如今接口上,類上和方法上
出現接口上,表示該接口的全部實現類都有事務支持
出如今類上,表示類中全部方法有事務支持
出如今方法上,表示方法有事務支持
例以下例中,咱們類中指定了事務的爲只讀型,可是下面的轉帳還涉及到了寫操做,因此又在方法上增長了一個 readOnly 值爲 false 的註解
@Service("accountService")
@Transactional(readOnly = true, propagation = Propagation.SUPPORTS)
public class AccountServiceImpl implements AccountService {
.... 省略
@Transactional(readOnly=false,propagation=Propagation.REQUIRED)
public void transfer(String sourceName, String targetName, Float money) {
...... 省略
}
}
複製代碼
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(locations = "classpath:bean.xml")
public class AccountServiceTest {
@Autowired
private AccountService as;
@Test
public void testTransfer() {
as.transfer("張三", "李四", 500f);
}
}
複製代碼
下面使用的就是純註解的方式,bean.xml 就能夠刪除掉了,這種方式不是很難
獲取容器時須要使用下列形式
private ApplicationContext ac = new AnnotationConfigApplicationContext(SpringConfiguration.class);
複製代碼
若是使用了 spring 的單元測試
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(classes= SpringConfiguration.class)
public class AccountServiceTest {
......
}
複製代碼
@Configuration 至關於已經幫咱們把 bean.xml 文件創立好了,按照咱們往常的步驟,應該指定掃描的包了,這也就是咱們這個註解的做用
指定 spring 在初始化容器時要掃描的包,在 XML 中至關於:
<!--開啓掃描-->
<context:component-scan base-package="cn.ideal"></context:component-scan>
複製代碼
其中 basePackages 用於指定掃描的包,和這個註解中value屬性的做用是一致的
之前在建立數據源的時候,都是直接把配置信息寫死了,若是想要使用 properties 進行內容的配置,在這時候就須要,使用 @PropertySource 這個註解
SpringConfiguration 類(至關於 bean.xml)
/** * Spring 配置類 */
@Configuration
@ComponentScan("cn.ideal")
@Import({JdbcConfig.class,TransactionConfig.class})
@PropertySource("jdbcConfig.properties")
@EnableTransactionManagement
public class SpringConfiguration {
}
複製代碼
寫好了配置類,以及指定了掃描的包,下面該作的就是配置 jdbcTemplate 以及數據源,再有就是建立事務管理器對象,在 XML 中咱們會經過書寫 bean 標籤來配置,而 Spring 爲咱們提供了 @Bean 這個註解來替代原來的標籤
JdbcConfig (JDBC配置類)
/** * 和鏈接數據庫相關的配置類 */
public class JdbcConfig {
@Value("${jdbc.driver}")
private String driver;
@Value("${jdbc.url}")
private String url;
@Value("${jdbc.username}")
private String username;
@Value("${jdbc.password}")
private String password;
/** * 建立JdbcTemplate * @param dataSource * @return */
@Bean(name="jdbcTemplate")
public JdbcTemplate createJdbcTemplate(DataSource dataSource){
return new JdbcTemplate(dataSource);
}
/** * 建立數據源對象 * @return */
@Bean(name="dataSource")
public DataSource createDataSource(){
DriverManagerDataSource ds = new DriverManagerDataSource();
ds.setDriverClassName(driver);
ds.setUrl(url);
ds.setUsername(username);
ds.setPassword(password);
return ds;
}
}
複製代碼
jdbcConfig.properties
將配置文件單獨配置出來
jdbc.driver=com.mysql.jdbc.Driver
jdbc.url=jdbc:mysql://localhost:3306/ideal_spring
jdbc.username=root
jdbc.password=root99
複製代碼
TransactionConfig
/** * 和事務相關的配置類 */
public class TransactionConfig {
/** * 用於建立事務管理器對象 * @param dataSource * @return */
@Bean(name="transactionManager")
public PlatformTransactionManager createTransactionManager(DataSource dataSource){
return new DataSourceTransactionManager(dataSource);
}
}
複製代碼
① 這篇文章就寫到這裏了,學習任何一門技術,只有知其然,才能明白其全部然,不少人在某個技術領域已經沉浸多年,天然有了特殊的思考與理解,憑藉着強大的經驗,天然也能快速上手,但若是處於門外狀態,或者對這一方面接觸的很少,就更須要了解一門技術的來龍去脈,不過什麼源碼分析,各類設計模式,這也都是後話,咱們的第一要義就是要用它作事,要讓他跑起來,自認爲我不是什麼過於聰明的人,直接去學習一堆配置,一堆註解,一堆專有名詞,太空洞了,很難理解。
② 咱們每每都陷入了一種,爲學而學的狀態,可能你們都會SSM我也學,你們都說 SpringBoot 簡單舒服,我也去學,固然不少時候由於一些工做或者學習的須要,沒有辦法,可是仍以爲,私下再次看一門技術的時候,能夠藉助一些文章或者資料,亦或者找點視頻資源,去看看這一門究竟帶來了什麼,其過人之處,必然是解決了咱們之前遇到的,或者沒考慮到的問題,這樣一種按部就班的學習方式,能夠幫助咱們對一些技術有一個總體的概念,以及瞭解其之間的聯繫。
③ 這一篇文章,我參考了 《Spring 實戰》、某馬的視頻、以及百度谷歌上的一些參考內容,從一個很是簡單的 增刪改查的案例出發,經過分析其事務問題,一步一步從動態代理,到 AOP進行了屢次的改進,其中涉及到一些例如 動態代理或者JdcbTemplate的知識,或許有的朋友不熟悉,我也用了一些篇幅說明,寫這樣一篇長文章,確實很費功夫,若是想要了解 Spring AOP 相關知識的朋友,能夠看一看,也能夠當作一個簡單的參考,用來手生的時候做爲工具書參考
很是但願能給你們帶來幫助,再次感謝你們的支持,謝謝!
Tips:同時有須要的朋友能夠去看個人前一篇文章
【萬字長文】Spring框架 層層遞進輕鬆入門 (IOC和DI)
若是文章中有什麼不足,歡迎你們留言交流,感謝朋友們的支持!
若是能幫到你的話,那就來關注我吧!若是您更喜歡微信文章的閱讀方式,能夠關注個人公衆號
在這裏的咱們素不相識,卻都在爲了本身的夢而努力 ❤
一個堅持推送原創開發技術文章的公衆號:理想二旬不止