Spring?AOP底層機(jī)制之代理模式
代理模式
為什么要學(xué)習(xí)代理模式,因為AOP的底層機(jī)制就是動態(tài)代理!
代理模式:
- 靜態(tài)代理
- 動態(tài)代理
學(xué)習(xí)aop之前 , 我們要先了解一下代理模式!

靜態(tài)代理
靜態(tài)代理角色分析
- 抽象角色 : 一般使用接口或者抽象類來實現(xiàn)
- 真實角色 : 被代理的角色
- 代理角色 : 代理真實角色 ; 代理真實角色后 , 一般會做一些附屬的操作 .
- 客戶 : 使用代理角色來進(jìn)行一些操作 .
代碼實現(xiàn)
Rent . java 即抽象角色
public interface Rent {
public void rent();
}Host . java 即真實角色
//真實角色: 房東,房東要出租房子
public class Host implements Rent{
public void rent() {
System.out.println("房屋出租");
}
}Proxy . java 即代理角色
//代理角色:中介
public class Proxy implements Rent {
private Host host;
public Proxy() { }
public Proxy(Host host) {
this.host = host;
}
//租房
public void rent(){
seeHouse();
host.rent();
fare();
}
//看房
public void seeHouse(){
System.out.println("帶房客看房");
}
//收中介費
public void fare(){
System.out.println("收中介費");
}
}Client . java 即客戶
//客戶類,一般客戶都會去找代理!
public class Client {
public static void main(String[] args) {
//房東要租房
Host host = new Host();
//中介幫助房東
Proxy proxy = new Proxy(host);
//你去找中介!
proxy.rent();
}
}分析:在這個過程中,你直接接觸的就是中介,就如同現(xiàn)實生活中的樣子,你看不到房東,但是你依舊租到了房東的房子通過代理,這就是所謂的代理模式,程序源自于生活,所以學(xué)編程的人,一般能夠更加抽象的看待生活中發(fā)生的事情。
靜態(tài)代理的好處:
- 可以使得我們的真實角色更加純粹 . 不再去關(guān)注一些公共的事情 .
- 公共的業(yè)務(wù)由代理來完成 . 實現(xiàn)了業(yè)務(wù)的分工 ,
- 公共業(yè)務(wù)發(fā)生擴(kuò)展時變得更加集中和方便 .
缺點 :
類多了 , 多了代理類 , 工作量變大了 . 開發(fā)效率降低 .
我們想要靜態(tài)代理的好處,又不想要靜態(tài)代理的缺點,所以 , 就有了動態(tài)代理 !
靜態(tài)代理再理解
同學(xué)們練習(xí)完畢后,我們再來舉一個例子,鞏固大家的學(xué)習(xí)!
練習(xí)步驟:
1、創(chuàng)建一個抽象角色,比如咋們平時做的用戶業(yè)務(wù),抽象起來就是增刪改查!
//抽象角色:增刪改查業(yè)務(wù)
public interface UserService {
void add();
void delete();
void update();
void query();
}2、我們需要一個真實對象來完成這些增刪改查操作
//真實對象,完成增刪改查操作的人
public class UserServiceImpl implements UserService {
public void add() {
System.out.println("增加了一個用戶");
}
public void delete() {
System.out.println("刪除了一個用戶");
}
public void update() {
System.out.println("更新了一個用戶");
}
public void query() {
System.out.println("查詢了一個用戶");
}
}3、需求來了,現(xiàn)在我們需要增加一個日志功能,怎么實現(xiàn)!
- 思路1 :在實現(xiàn)類上增加代碼 【麻煩!】
- 思路2:使用代理來做,能夠不改變原來的業(yè)務(wù)情況下,實現(xiàn)此功能就是最好的了!
4、設(shè)置一個代理類來處理日志!代理角色
//代理角色,在這里面增加日志的實現(xiàn)
public class UserServiceProxy implements UserService {
private UserServiceImpl userService;
public void setUserService(UserServiceImpl userService) {
this.userService = userService;
}
public void add() {
log("add");
userService.add();
}
public void delete() {
log("delete");
userService.delete();
}
public void update() {
log("update");
userService.update();
}
public void query() {
log("query");
userService.query();
}
public void log(String msg){
System.out.println("執(zhí)行了"+msg+"方法");
}
}5、測試訪問類:
public class Client {
public static void main(String[] args) {
//真實業(yè)務(wù)
UserServiceImpl userService = new UserServiceImpl();
//代理類
UserServiceProxy proxy = new UserServiceProxy();
//使用代理類實現(xiàn)日志功能!
proxy.setUserService(userService);
proxy.add();
}
}OK,到了現(xiàn)在代理模式大家應(yīng)該都沒有什么問題了,重點大家需要理解其中的思想;
我們在不改變原來的代碼的情況下,實現(xiàn)了對原有功能的增強(qiáng),這是AOP中最核心的思想
聊聊AOP:縱向開發(fā),橫向開發(fā)

動態(tài)代理
動態(tài)代理的角色和靜態(tài)代理的一樣 .
動態(tài)代理的代理類是動態(tài)生成的 . 靜態(tài)代理的代理類是我們提前寫好的
動態(tài)代理分為兩類 : 一類是基于接口動態(tài)代理 , 一類是基于類的動態(tài)代理
- 基于接口的動態(tài)代理----JDK動態(tài)代理
- 基于類的動態(tài)代理–cglib
- 現(xiàn)在用的比較多的是 javasist 來生成動態(tài)代理 . 百度一下javasist
- 我們這里使用JDK的原生代碼來實現(xiàn),其余的道理都是一樣的!
JDK的動態(tài)代理需要了解兩個類
核心 : InvocationHandler 和 Proxy , 打開JDK幫助文檔看看
【InvocationHandler:調(diào)用處理程序】

Object invoke(Object proxy, 方法 method, Object[] args); //參數(shù) //proxy - 調(diào)用該方法的代理實例 //method -所述方法對應(yīng)于調(diào)用代理實例上的接口方法的實例。方法對象的聲明類將是該方法聲明的接口,它可以是代理類繼承該方法的代理接口的超級接口。 //args -包含的方法調(diào)用傳遞代理實例的參數(shù)值的對象的陣列,或null如果接口方法沒有參數(shù)。原始類型的參數(shù)包含在適當(dāng)?shù)脑及b器類的實例中,例如java.lang.Integer或java.lang.Boolean 。
【Proxy : 代理】



//生成代理類
public Object getProxy(){
return Proxy.newProxyInstance(this.getClass().getClassLoader(),
rent.getClass().getInterfaces(),this);
}代碼實現(xiàn)
抽象角色和真實角色和之前的一樣!
Rent . java 即抽象角色
//抽象角色:租房
public interface Rent {
public void rent();
}Host . java 即真實角色
//真實角色: 房東,房東要出租房子
public class Host implements Rent{
public void rent() {
System.out.println("房屋出租");
}
}ProxyInvocationHandler. java 即代理角色
public class ProxyInvocationHandler implements InvocationHandler {
private Rent rent;
public void setRent(Rent rent) {
this.rent = rent;
}
//生成代理類,重點是第二個參數(shù),獲取要代理的抽象角色!之前都是一個角色,現(xiàn)在可以代理一類角色
public Object getProxy(){
return Proxy.newProxyInstance(this.getClass().getClassLoader(),
rent.getClass().getInterfaces(),this);
}
// proxy : 代理類 method : 代理類的調(diào)用處理程序的方法對象.
// 處理代理實例上的方法調(diào)用并返回結(jié)果
@Override
public Object invoke(Object proxy, Method method, Object[] args) throwsThrowable {
seeHouse();
//核心:本質(zhì)利用反射實現(xiàn)!
Object result = method.invoke(rent, args);
fare();
return result;
}
//看房
public void seeHouse(){
System.out.println("帶房客看房");
}
//收中介費
public void fare(){
System.out.println("收中介費");
}
}Client . java
//租客
public class Client {
public static void main(String[] args) {
//真實角色
Host host = new Host();
//代理實例的調(diào)用處理程序
ProxyInvocationHandler pih = new ProxyInvocationHandler();
pih.setRent(host); //將真實角色放置進(jìn)去!
Rent proxy = (Rent)pih.getProxy(); //動態(tài)生成對應(yīng)的代理類!
proxy.rent();
}
}核心:一個動態(tài)代理 , 一般代理某一類業(yè)務(wù) , 一個動態(tài)代理可以代理多個類,代理的是接口!、
深化理解
我們來使用動態(tài)代理實現(xiàn)代理我們后面寫的UserService!
我們也可以編寫一個通用的動態(tài)代理實現(xiàn)的類!所有的代理對象設(shè)置為Object即可!
public class ProxyInvocationHandler implements InvocationHandler {
private Object target;
public void setTarget(Object target) {
this.target = target;
}
//生成代理類
public Object getProxy(){
return Proxy.newProxyInstance(this.getClass().getClassLoader(),
target.getClass().getInterfaces(),this);
}
// proxy : 代理類
// method : 代理類的調(diào)用處理程序的方法對象.
public Object invoke(Object proxy, Method method, Object[] args) throwsThrowable {
log(method.getName());
Object result = method.invoke(target, args);
return result;
}
public void log(String methodName){
System.out.println("執(zhí)行了"+methodName+"方法");
}
}測試!
public class Test {
public static void main(String[] args) {
//真實對象
UserServiceImpl userService = new UserServiceImpl();
//代理對象的調(diào)用處理程序
ProxyInvocationHandler pih = new ProxyInvocationHandler();
pih.setTarget(userService); //設(shè)置要代理的對象
UserService proxy = (UserService)pih.getProxy(); //動態(tài)生成代理類!
proxy.delete();
}
}測試,增刪改查,查看結(jié)果!
動態(tài)代理的好處
靜態(tài)代理有的它都有,靜態(tài)代理沒有的,它也有!
- 可以使得我們的真實角色更加純粹 . 不再去關(guān)注一些公共的事情
- 公共的業(yè)務(wù)由代理來完成 . 實現(xiàn)了業(yè)務(wù)的分工
- 公共業(yè)務(wù)發(fā)生擴(kuò)展時變得更加集中和方便
- 一個動態(tài)代理 , 一般代理某一類業(yè)務(wù)
- 一個動態(tài)代理可以代理多個類,代理的是接口!
到此這篇關(guān)于Spring AOP底層機(jī)制之代理模式的文章就介紹到這了,更多相關(guān)Spring代理模式內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
SpringBoot整合RestTemplate用法的實現(xiàn)
本篇主要介紹了RestTemplate中的GET,POST,PUT,DELETE、文件上傳和文件下載6大常用的功能,具有一定的參考價值,感興趣的可以了解一下2023-08-08
POST方法給@RequestBody傳參數(shù)失敗的解決及原因分析
這篇文章主要介紹了POST方法給@RequestBody傳參數(shù)失敗的解決及原因分析,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-10-10
Spring Boot中實現(xiàn)定時任務(wù)應(yīng)用實踐
定時任務(wù)一般是項目中都需要用到的,可以用于定時處理一些特殊的任務(wù)。下面這篇文章主要給大家介紹了關(guān)于Spring Boot中實現(xiàn)定時任務(wù)應(yīng)用實踐的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),需要的朋友可以參考下。2018-05-05

