Spring?AOP原理及動態(tài)代理
一、什么是代理?
指為一個目標(biāo)對象提供一個代理對象, 并由代理對象控制對目標(biāo)對象的引用. 使用代理對象, 是為了在不修改目標(biāo)對象的基礎(chǔ)上,增強(qiáng)目標(biāo)對象的業(yè)務(wù)邏輯.

1、靜態(tài)代理
靜態(tài)代理的特點(diǎn)是, 為每一個業(yè)務(wù)增強(qiáng)都提供一個代理類, 由代理類來創(chuàng)建代理對象. 下面我們通過靜態(tài)代理來實(shí)現(xiàn)對轉(zhuǎn)賬業(yè)務(wù)進(jìn)行身份驗證.
(1) 轉(zhuǎn)賬業(yè)務(wù)
public interface IAccountService {
//主業(yè)務(wù)邏輯: 轉(zhuǎn)賬
void transfer();
}
public class AccountServiceImpl implements IAccountService {
@Override
public void transfer() {
System.out.println("調(diào)用dao層,完成轉(zhuǎn)賬主業(yè)務(wù).");
}
}
(2) 代理類
public class AccountProxy implements IAccountService {
//目標(biāo)對象
private IAccountService target;
public AccountProxy(IAccountService target) {
this.target = target;
}
/**
* 代理方法,實(shí)現(xiàn)對目標(biāo)方法的功能增強(qiáng)
*/
@Override
public void transfer() {
before();
target.transfer();
}
/**
* 前置增強(qiáng)
*/
private void before() {
System.out.println("對轉(zhuǎn)賬人身份進(jìn)行驗證.");
}
}(3) 測試
public class Client {
public static void main(String[] args) {
//創(chuàng)建目標(biāo)對象
IAccountService target = new AccountServiceImpl();
//創(chuàng)建代理對象
AccountProxy proxy = new AccountProxy(target);
proxy.transfer();
}
}結(jié)果: 對轉(zhuǎn)賬人身份進(jìn)行驗證.調(diào)用dao層,完成轉(zhuǎn)賬主業(yè)務(wù).
2、動態(tài)代理
靜態(tài)代理會為每一個業(yè)務(wù)增強(qiáng)都提供一個代理類, 由代理類來創(chuàng)建代理對象, 而動態(tài)代理并不存在代理類, 代理對象直接由代理生成工具動態(tài)生成.
2.1、JDK動態(tài)代理
JDK動態(tài)代理是使用 java.lang.reflect 包下的代理類來實(shí)現(xiàn). JDK動態(tài)代理動態(tài)代理必須要有接口.
(1) 轉(zhuǎn)賬業(yè)務(wù)
public interface IAccountService {
//主業(yè)務(wù)邏輯: 轉(zhuǎn)賬
void transfer();
}
public class AccountServiceImpl implements IAccountService {
@Override
public void transfer() {
System.out.println("調(diào)用dao層,完成轉(zhuǎn)賬主業(yè)務(wù).");
}
}(2) 增強(qiáng)
因為這里沒有配置切入點(diǎn), 稱為切面會有點(diǎn)奇怪, 所以稱為增強(qiáng).
public class AccountAdvice implements InvocationHandler {
//目標(biāo)對象
private IAccountService target;
public AccountAdvice(IAccountService target) {
this.target = target;
}
/**
* 代理方法, 每次調(diào)用目標(biāo)方法時都會進(jìn)到這里
*/
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
before();
return method.invoke(target, args);
}
/**
* 前置增強(qiáng)
*/
private void before() {
System.out.println("對轉(zhuǎn)賬人身份進(jìn)行驗證.");
}
}(3) 測試
public class Client {
public static void main(String[] args) {
//創(chuàng)建目標(biāo)對象
IAccountService target = new AccountServiceImpl();
//創(chuàng)建代理對象
IAccountService proxy = (IAccountService) Proxy.newProxyInstance(target.getClass().getClassLoader(),
target.getClass().getInterfaces(),
new AccountAdvice(target)
);
proxy.transfer();
}
}結(jié)果: 對轉(zhuǎn)賬人身份進(jìn)行驗證.調(diào)用dao層,完成轉(zhuǎn)賬主業(yè)務(wù).
2.2、 CGLIB動態(tài)代理
JDK動態(tài)代理必須要有接口, 但如果要代理一個沒有接口的類該怎么辦呢? 這時我們可以使用CGLIB動態(tài)代理. CGLIB動態(tài)代理的原理是生成目標(biāo)類的子類, 這個子類對象就是代理對象, 代理對象是被增強(qiáng)過的.
注意: 不管有沒有接口都可以使用CGLIB動態(tài)代理, 而不是只有在無接口的情況下才能使用
(1) 轉(zhuǎn)賬業(yè)務(wù)
public class AccountService {
public void transfer() {
System.out.println("調(diào)用dao層,完成轉(zhuǎn)賬主業(yè)務(wù).");
}
}(2) 增強(qiáng)
因為這里沒有配置切入點(diǎn), 稱為切面會有點(diǎn)奇怪, 所以稱為增強(qiáng).
public class AccountAdvice implements MethodInterceptor {
/**
* 代理方法, 每次調(diào)用目標(biāo)方法時都會進(jìn)到這里
*/
@Override
public Object intercept(Object obj, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {
before();
return methodProxy.invokeSuper(obj, args);
// return method.invoke(obj, args); 這種也行
}
/**
* 前置增強(qiáng)
*/
private void before() {
System.out.println("對轉(zhuǎn)賬人身份進(jìn)行驗證.");
}
}(3) 測試
public class Client {
public static void main(String[] args) {
//創(chuàng)建目標(biāo)對象
AccountService target = new AccountService();
//
//創(chuàng)建代理對象
AccountService proxy = (AccountService) Enhancer.create(target.getClass(),
new AccountAdvice());
proxy.transfer();
}
}結(jié)果: 對轉(zhuǎn)賬人身份進(jìn)行驗證.調(diào)用dao層,完成轉(zhuǎn)賬主業(yè)務(wù).
二、模擬Spring AOP場景
了解了動態(tài)代理后, 我們就可以自己來實(shí)現(xiàn)Spring AOP功能了, 所以下面我們來模擬下Spring AOP場景.
(1) 轉(zhuǎn)賬業(yè)務(wù)
public interface IAccountService {
//主業(yè)務(wù)邏輯: 轉(zhuǎn)賬
void transfer();
}
public class AccountServiceImpl implements IAccountService {
@Override
public void transfer() {
System.out.println("調(diào)用dao層,完成轉(zhuǎn)賬主業(yè)務(wù).");
}
}(2) 切面抽象類
定義一個切面抽象類, 該類使用了模板方法的設(shè)計模式, 為開始, 結(jié)束, 異常, 前置增強(qiáng), 后置增強(qiáng)提供了默認(rèn)實(shí)現(xiàn), 當(dāng)我們定義切面類時只需要按需重寫它們就行. isIntercept() 方法用來判斷切入點(diǎn)是否正確, 切面類需要重寫這個方法.
public abstract class BaseAspect implements MethodInterceptor {
private static final Logger logger = LoggerFactory.getLogger(BaseAspect.class);
@Override
public Object intercept(Object obj, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {
Object result = null;
begin();
try {
if (isIntercept(method, args)) {
before();
result = methodProxy.invokeSuper(obj, args);
after();
} else {
result = methodProxy.invokeSuper(obj,args);
}
} catch (Exception e) {
logger.error("proxy failure", e);
error(e);
throw e;
} finally {
end();
}
return result;
}
/**
* 開始增強(qiáng)
*/
public void begin() {
}
/**
* 切入點(diǎn)判斷
*/
public boolean isIntercept(Method method, Object[] args) throws Throwable {
return true;
}
/**
* 前置增強(qiáng)
*/
public void before() throws Throwable {
}
/**
* 后置增強(qiáng)
*/
public void after() throws Throwable {
}
/**
* 異常增強(qiáng)
*/
public void error(Throwable e) {
}
/**
* 最終增強(qiáng)
*/
public void end() {
}
}(3) 切面類
創(chuàng)建一個切面類, 類中配置切入點(diǎn)和增強(qiáng).
public class AccountAspect extends BaseAspect {
/**
* 切入點(diǎn)
*/
public boolean isIntercept(Method method, Object[] args) throws Throwable {
return method.getName().equals("transfer");
}
/**
* 前置增強(qiáng)
*/
public void before() throws Throwable {
System.out.println("對轉(zhuǎn)賬人身份進(jìn)行驗證.");
}
}(4) 代理工廠類
定義一個工廠類來創(chuàng)建代理, 其實(shí)不創(chuàng)建這個類也行, 但為了模仿Spring還是創(chuàng)建了. @SuppressWarnings是為了抑制警告, 就是編譯器上面的黃線.
public class ProxyFactory {
@SuppressWarnings("unchecked")
public static <T> T createProxy(final Class<?> targetClass, final MethodInterceptor methodInterceptor) {
return (T) Enhancer.create(targetClass,methodInterceptor);
}
}(5) 測試
public class Client {
public static void main(String[] args) {
//創(chuàng)建目標(biāo)對象
IAccountService target = new AccountServiceImpl();
//切面
BaseAspect accountAspect = new AccountAspect();
//創(chuàng)建代理對象
IAccountService proxy = (IAccountService) ProxyFactory.createProxy(target.getClass(), accountAspect);
proxy.transfer();
}
}結(jié)果:對轉(zhuǎn)賬人身份進(jìn)行驗證.調(diào)用dao層,完成轉(zhuǎn)賬主業(yè)務(wù).
到此這篇關(guān)于Spring AOP原理及動態(tài)代理的文章就介紹到這了,更多相關(guān)Spring AOP 內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
maven利用tomcat插件部署遠(yuǎn)程Linux服務(wù)器的步驟詳解
Maven已經(jīng)是Java的項目管理常用方式,下面這篇文章主要給大家介紹了關(guān)于maven利用tomcat插件部署遠(yuǎn)程Linux服務(wù)器的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),需要的朋友可以參考借鑒,下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧。2017-11-11
Java編程通過list接口實(shí)現(xiàn)數(shù)據(jù)的增刪改查代碼示例
這篇文章是介紹Java編程基礎(chǔ)方面的內(nèi)容,涉及l(fā)ist接口的操作,通過list接口實(shí)現(xiàn)對數(shù)據(jù)的增刪改查的相關(guān)代碼,具有一定參考價值,需要的朋友可以了解下。2017-10-10
java wait()/notify() 實(shí)現(xiàn)生產(chǎn)者消費(fèi)者模式詳解
這篇文章主要介紹了java wait()/notify() 實(shí)現(xiàn)生產(chǎn)者消費(fèi)者模式詳解,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-07-07
java&javascript自定義加密數(shù)據(jù)傳輸代碼示例
這篇文章主要介紹了java&javascript自定義加密數(shù)據(jù)傳輸代碼示例,具有一定參考價值,需要的朋友可以了解下。2017-11-11
Java使用wait/notify實(shí)現(xiàn)線程間通信下篇
wait()和notify()是直接隸屬于Object類,也就是說所有對象都擁有這一對方法,下面這篇文章主要給大家介紹了關(guān)于使用wait/notify實(shí)現(xiàn)線程間通信的相關(guān)資料,需要的朋友可以參考下2022-12-12
解決Spring Boot和Feign中使用Java 8時間日期API(LocalDate等)的序列化問題
這篇文章主要介紹了解決Spring Boot和Feign中使用Java 8時間日期API(LocalDate等)的序列化問題,需要的朋友可以參考下2018-03-03

