您好,登錄后才能下訂單哦!
這篇文章主要介紹“Spring AOP底層機(jī)制之代理模式怎么實(shí)現(xiàn)”,在日常操作中,相信很多人在Spring AOP底層機(jī)制之代理模式怎么實(shí)現(xiàn)問題上存在疑惑,小編查閱了各式資料,整理出簡(jiǎn)單好用的操作方法,希望對(duì)大家解答”Spring AOP底層機(jī)制之代理模式怎么實(shí)現(xiàn)”的疑惑有所幫助!接下來,請(qǐng)跟著小編一起來學(xué)習(xí)吧!
為什么要學(xué)習(xí)代理模式,因?yàn)锳OP的底層機(jī)制就是動(dòng)態(tài)代理!
代理模式:
靜態(tài)代理
動(dòng)態(tài)代理
學(xué)習(xí)aop之前 , 我們要先了解一下代理模式!
靜態(tài)代理角色分析
抽象角色 : 一般使用接口或者抽象類來實(shí)現(xiàn)
真實(shí)角色 : 被代理的角色
代理角色 : 代理真實(shí)角色 ; 代理真實(shí)角色后 , 一般會(huì)做一些附屬的操作 .
客戶 : 使用代理角色來進(jìn)行一些操作 .
代碼實(shí)現(xiàn)
Rent . java 即抽象角色
public interface Rent { public void rent(); }
Host . java 即真實(shí)角色
//真實(shí)角色: 房東,房東要出租房子 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("帶房客看房"); } //收中介費(fèi) public void fare(){ System.out.println("收中介費(fèi)"); } }
Client . java 即客戶
//客戶類,一般客戶都會(huì)去找代理! public class Client { public static void main(String[] args) { //房東要租房 Host host = new Host(); //中介幫助房東 Proxy proxy = new Proxy(host); //你去找中介! proxy.rent(); } }
分析:在這個(gè)過程中,你直接接觸的就是中介,就如同現(xiàn)實(shí)生活中的樣子,你看不到房東,但是你依舊租到了房東的房子通過代理,這就是所謂的代理模式,程序源自于生活,所以學(xué)編程的人,一般能夠更加抽象的看待生活中發(fā)生的事情。
靜態(tài)代理的好處:
可以使得我們的真實(shí)角色更加純粹 . 不再去關(guān)注一些公共的事情 .
公共的業(yè)務(wù)由代理來完成 . 實(shí)現(xiàn)了業(yè)務(wù)的分工 ,
公共業(yè)務(wù)發(fā)生擴(kuò)展時(shí)變得更加集中和方便 .
缺點(diǎn) :
類多了 , 多了代理類 , 工作量變大了 . 開發(fā)效率降低 .
我們想要靜態(tài)代理的好處,又不想要靜態(tài)代理的缺點(diǎn),所以 , 就有了動(dòng)態(tài)代理 !
同學(xué)們練習(xí)完畢后,我們?cè)賮砼e一個(gè)例子,鞏固大家的學(xué)習(xí)!
練習(xí)步驟:
1、創(chuàng)建一個(gè)抽象角色,比如咋們平時(shí)做的用戶業(yè)務(wù),抽象起來就是增刪改查!
//抽象角色:增刪改查業(yè)務(wù) public interface UserService { void add(); void delete(); void update(); void query(); }
2、我們需要一個(gè)真實(shí)對(duì)象來完成這些增刪改查操作
//真實(shí)對(duì)象,完成增刪改查操作的人 public class UserServiceImpl implements UserService { public void add() { System.out.println("增加了一個(gè)用戶"); } public void delete() { System.out.println("刪除了一個(gè)用戶"); } public void update() { System.out.println("更新了一個(gè)用戶"); } public void query() { System.out.println("查詢了一個(gè)用戶"); } }
3、需求來了,現(xiàn)在我們需要增加一個(gè)日志功能,怎么實(shí)現(xiàn)!
思路1 :在實(shí)現(xiàn)類上增加代碼 【麻煩!】
思路2:使用代理來做,能夠不改變?cè)瓉淼臉I(yè)務(wù)情況下,實(shí)現(xiàn)此功能就是最好的了!
4、設(shè)置一個(gè)代理類來處理日志!代理角色
//代理角色,在這里面增加日志的實(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、測(cè)試訪問類:
public class Client { public static void main(String[] args) { //真實(shí)業(yè)務(wù) UserServiceImpl userService = new UserServiceImpl(); //代理類 UserServiceProxy proxy = new UserServiceProxy(); //使用代理類實(shí)現(xiàn)日志功能! proxy.setUserService(userService); proxy.add(); } }
OK,到了現(xiàn)在代理模式大家應(yīng)該都沒有什么問題了,重點(diǎn)大家需要理解其中的思想;
我們?cè)诓桓淖冊(cè)瓉淼拇a的情況下,實(shí)現(xiàn)了對(duì)原有功能的增強(qiáng),這是AOP中最核心的思想
聊聊AOP:縱向開發(fā),橫向開發(fā)
動(dòng)態(tài)代理的角色和靜態(tài)代理的一樣 .
動(dòng)態(tài)代理的代理類是動(dòng)態(tài)生成的 . 靜態(tài)代理的代理類是我們提前寫好的
動(dòng)態(tài)代理分為兩類 : 一類是基于接口動(dòng)態(tài)代理 , 一類是基于類的動(dòng)態(tài)代理
基于接口的動(dòng)態(tài)代理----JDK動(dòng)態(tài)代理
基于類的動(dòng)態(tài)代理–cglib
現(xiàn)在用的比較多的是 javasist 來生成動(dòng)態(tài)代理 . 百度一下javasist
我們這里使用JDK的原生代碼來實(shí)現(xiàn),其余的道理都是一樣的!
JDK的動(dòng)態(tài)代理需要了解兩個(gè)類
核心 : InvocationHandler 和 Proxy , 打開JDK幫助文檔看看
【InvocationHandler:調(diào)用處理程序】
Object invoke(Object proxy, 方法 method, Object[] args); //參數(shù) //proxy - 調(diào)用該方法的代理實(shí)例 //method -所述方法對(duì)應(yīng)于調(diào)用代理實(shí)例上的接口方法的實(shí)例。方法對(duì)象的聲明類將是該方法聲明的接口,它可以是代理類繼承該方法的代理接口的超級(jí)接口。 //args -包含的方法調(diào)用傳遞代理實(shí)例的參數(shù)值的對(duì)象的陣列,或null如果接口方法沒有參數(shù)。原始類型的參數(shù)包含在適當(dāng)?shù)脑及b器類的實(shí)例中,例如java.lang.Integer或java.lang.Boolean 。
【Proxy : 代理】
//生成代理類 public Object getProxy(){ return Proxy.newProxyInstance(this.getClass().getClassLoader(), rent.getClass().getInterfaces(),this); }
代碼實(shí)現(xiàn)
抽象角色和真實(shí)角色和之前的一樣!
Rent . java 即抽象角色
//抽象角色:租房 public interface Rent { public void rent(); }
Host . java 即真實(shí)角色
//真實(shí)角色: 房東,房東要出租房子 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; } //生成代理類,重點(diǎn)是第二個(gè)參數(shù),獲取要代理的抽象角色!之前都是一個(gè)角色,現(xiàn)在可以代理一類角色 public Object getProxy(){ return Proxy.newProxyInstance(this.getClass().getClassLoader(), rent.getClass().getInterfaces(),this); } // proxy : 代理類 method : 代理類的調(diào)用處理程序的方法對(duì)象. // 處理代理實(shí)例上的方法調(diào)用并返回結(jié)果 @Override public Object invoke(Object proxy, Method method, Object[] args) throwsThrowable { seeHouse(); //核心:本質(zhì)利用反射實(shí)現(xiàn)! Object result = method.invoke(rent, args); fare(); return result; } //看房 public void seeHouse(){ System.out.println("帶房客看房"); } //收中介費(fèi) public void fare(){ System.out.println("收中介費(fèi)"); } }
Client . java
//租客 public class Client { public static void main(String[] args) { //真實(shí)角色 Host host = new Host(); //代理實(shí)例的調(diào)用處理程序 ProxyInvocationHandler pih = new ProxyInvocationHandler(); pih.setRent(host); //將真實(shí)角色放置進(jìn)去! Rent proxy = (Rent)pih.getProxy(); //動(dòng)態(tài)生成對(duì)應(yīng)的代理類! proxy.rent(); } }
核心:一個(gè)動(dòng)態(tài)代理 , 一般代理某一類業(yè)務(wù) , 一個(gè)動(dòng)態(tài)代理可以代理多個(gè)類,代理的是接口!、
我們來使用動(dòng)態(tài)代理實(shí)現(xiàn)代理我們后面寫的UserService!
我們也可以編寫一個(gè)通用的動(dòng)態(tài)代理實(shí)現(xiàn)的類!所有的代理對(duì)象設(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)用處理程序的方法對(duì)象. 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+"方法"); } }
測(cè)試!
public class Test { public static void main(String[] args) { //真實(shí)對(duì)象 UserServiceImpl userService = new UserServiceImpl(); //代理對(duì)象的調(diào)用處理程序 ProxyInvocationHandler pih = new ProxyInvocationHandler(); pih.setTarget(userService); //設(shè)置要代理的對(duì)象 UserService proxy = (UserService)pih.getProxy(); //動(dòng)態(tài)生成代理類! proxy.delete(); } }
測(cè)試,增刪改查,查看結(jié)果!
靜態(tài)代理有的它都有,靜態(tài)代理沒有的,它也有!
可以使得我們的真實(shí)角色更加純粹 . 不再去關(guān)注一些公共的事情
公共的業(yè)務(wù)由代理來完成 . 實(shí)現(xiàn)了業(yè)務(wù)的分工
公共業(yè)務(wù)發(fā)生擴(kuò)展時(shí)變得更加集中和方便
一個(gè)動(dòng)態(tài)代理 , 一般代理某一類業(yè)務(wù)
一個(gè)動(dòng)態(tài)代理可以代理多個(gè)類,代理的是接口!
到此,關(guān)于“Spring AOP底層機(jī)制之代理模式怎么實(shí)現(xiàn)”的學(xué)習(xí)就結(jié)束了,希望能夠解決大家的疑惑。理論與實(shí)踐的搭配能更好的幫助大家學(xué)習(xí),快去試試吧!若想繼續(xù)學(xué)習(xí)更多相關(guān)知識(shí),請(qǐng)繼續(xù)關(guān)注億速云網(wǎng)站,小編會(huì)繼續(xù)努力為大家?guī)砀鄬?shí)用的文章!
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。