代理(proxy)是一種設(shè)計(jì)模式,提供了對目標(biāo)對象另外的訪問方式;即通過代理對象訪問目標(biāo)對象.這樣做的好處是:可以在目標(biāo)對象實(shí)現(xiàn)的基礎(chǔ)上,增強(qiáng)額外的功能操作,即擴(kuò)展目標(biāo)對象的功能。
這里使用到編程中的一個(gè)思想:不要隨意去修改別人已經(jīng)寫好的代碼或者方法,如果需改修改,可以通過代理的方式來擴(kuò)展該方法。
舉個(gè)例子來說明代理的作用:假設(shè)我們想邀請一位明星,那么并不是直接連接明星,而是聯(lián)系明星的經(jīng)紀(jì)人,來達(dá)到同樣的目的.明星就是一個(gè)目標(biāo)對象,他只要負(fù)責(zé)活動(dòng)中的節(jié)目,而其他瑣碎的事情就交給他的代理人(經(jīng)紀(jì)人)來解決.這就是代理思想在現(xiàn)實(shí)中的一個(gè)例子。
代理模式的 關(guān)鍵點(diǎn)是:代理對象與目標(biāo)對象.代理對象是對目標(biāo)對象的擴(kuò)展,并會(huì)調(diào)用目標(biāo)對象 .
靜態(tài)代理
靜態(tài)代理在使用時(shí),需要定義接口或者父類,被代理對象與代理對象一起實(shí)現(xiàn)相同的接口或者是繼承相同父類。
實(shí)例說明:
模擬保存動(dòng)作,定義一個(gè)保存動(dòng)作的接口:iuserdao.java,然后目標(biāo)對象實(shí)現(xiàn)這個(gè)接口的方法userdao.java,此時(shí)如果使用靜態(tài)代理方式,就需要在代理對象(userdaoproxy.java)中也實(shí)現(xiàn)iuserdao接口.調(diào)用的時(shí)候通過調(diào)用代理對象的方法來調(diào)用目標(biāo)對象。
需要 注意 的是,代理對象與目標(biāo)對象要實(shí)現(xiàn)相同的接口,然后通過調(diào)用相同的方法來調(diào)用目標(biāo)對象的方法。
接口:iuserdao.java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
package net.ydstudio.service; /** * @author nick * @projectname javalean * @package net.ydstudio.service * @createdate 2018/08/16 15:35 * @updatedate 2018/08/16 15:35 */ public interface iuserdao { /** * 保存數(shù)據(jù)庫 * @param: [] * @return: void */ void save(); } |
目標(biāo)對象:userdao.java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
|
package net.ydstudio.service.impl; import net.ydstudio.service.iuserdao; /** * @author nick * @projectname javalean * @package net.ydstudio.service.impl * @createdate 2018/08/16 15:36 * @updatedate 2018/08/16 15:36 */ public class userdao implements iuserdao { /** * 保存數(shù)據(jù)庫 * * @param: [] * @return: void */ public void save() { system.out.println( "數(shù)據(jù)已經(jīng)保存到數(shù)據(jù)庫" ); } } |
代理對象:userdaoproxy.java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
|
package net.ydstudio.staticproxy; import net.ydstudio.service.iuserdao; import net.ydstudio.service.impl.userdao; /** * @author nick * @projectname javalean * @package net.ydstudio.staticproxy * @createdate 2018/08/16 15:37 * @updatedate 2018/08/16 15:37 */ public class userdaoproxy implements iuserdao { /** * 保存被代理的對象 */ private userdao target; public userdaoproxy(userdao target) { this .target = target; } /** * 保存數(shù)據(jù)庫 * * @param: [] * @return: void */ public void save() { system.out.println( "開始保存數(shù)據(jù)……" ); target.save(); system.out.println( "結(jié)束保存數(shù)據(jù)……" ); } } |
測試類:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
|
package net.ydstudio.proxy; import net.ydstudio.service.iuserdao; import net.ydstudio.service.impl.userdao; import org.junit.test; import org.junit.runner.runwith; import org.junit.runners.junit4; import static org.junit. assert .*; /** * @author nick * @projectname javalean * @package net.ydstudio.proxy * @createdate 2018/08/16 15:58 * @updatedate 2018/08/16 15:58 */ @runwith (junit4. class ) public class proxyfactorytest { @test public void test(){ // 目標(biāo)對象 iuserdao target = new userdao(); system.out.println(target.getclass()); // 給目標(biāo)對象,創(chuàng)建代理對象 iuserdao proxy = (iuserdao) new proxyfactory(target).getproxyinstance(); // class $proxy()內(nèi)存中動(dòng)態(tài)生成的代理對象 system.out.println(proxy.getclass()); // 執(zhí)行方法 代理對象 proxy.save(); } } |
靜態(tài)代理總結(jié):
- 可以在不修改代理目標(biāo)對象的前提下,對代理目標(biāo)的功能進(jìn)行拓展。
- 需要實(shí)現(xiàn)代理目標(biāo)對象實(shí)現(xiàn)的接口,一旦代理目標(biāo)所實(shí)現(xiàn)的接口有修改,目標(biāo)對象與代理都需要維護(hù)。
要解決上面靜態(tài)代理的缺點(diǎn),就必須使用動(dòng)態(tài)代理的方式。
動(dòng)態(tài)代理
動(dòng)態(tài)代理有以下特點(diǎn):
- 代理對象,不需要實(shí)現(xiàn)接口
- 代理對象的生成,是利用jdk的api,動(dòng)態(tài)的在內(nèi)存中構(gòu)建代理對象(需要我們指定創(chuàng)建代理對象/目標(biāo)對象實(shí)現(xiàn)的接口的類型)
- 動(dòng)態(tài)代理也叫做:jdk代理,接口代理
jdk中生成代理對象的api
jdk實(shí)現(xiàn)代理只需要使用靜態(tài)的newproxyinstance方法,該方法需要接收三個(gè)參數(shù):
1
2
3
4
|
public static object newproxyinstance(classloader loader, class <?>[] interfaces, invocationhandler h) throws illegalargumentexception |
參數(shù)按順序解釋如下:
- classloader loader,:指定當(dāng)前目標(biāo)對象使用類加載器,獲取加載器的方法是固定的
- class[] interfaces,:目標(biāo)對象實(shí)現(xiàn)的接口的類型,使用泛型方式確認(rèn)類型
- invocationhandler h:事件處理,執(zhí)行目標(biāo)對象的方法時(shí),會(huì)觸發(fā)事件處理器的方法,會(huì)把當(dāng)前執(zhí)行目標(biāo)對象的方法作為參數(shù)傳入
代碼實(shí)現(xiàn):
接口類iuserdao.java以及接口實(shí)現(xiàn)類,目標(biāo)對象userdao是一樣的,沒有做修改.在這個(gè)基礎(chǔ)上,增加一個(gè)代理工廠類(proxyfactory.java),將代理類寫在這個(gè)地方,然后在測試類(需要使用到代理的代碼)中先建立目標(biāo)對象和代理對象的聯(lián)系,然后代用代理對象的中同名方法。
代理工廠類proxyfactory:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
|
package net.ydstudio.proxy; import java.lang.reflect.invocationhandler; import java.lang.reflect.method; import java.lang.reflect.proxy; /** * @author nick * @projectname javalean * @package net.ydstudio.proxy * @createdate 2018/08/16 15:44 * @updatedate 2018/08/16 15:44 */ public class proxyfactory { /** * 維護(hù)一個(gè)代理的目標(biāo)對象 */ private object target; public proxyfactory(object target){ this .target = target; } /** * 給目標(biāo)對象生成代理對象 * @param: [] * @return: java.lang.object */ public object getproxyinstance(){ return proxy.newproxyinstance( target.getclass().getclassloader(), target.getclass().getinterfaces(), new invocationhandler() { public object invoke(object proxy, method method, object[] args) throws throwable { system.out.println( "開始事務(wù)2" ); //執(zhí)行目標(biāo)對象方法 object returnvalue = method.invoke(target, args); system.out.println( "提交事務(wù)2" ); return returnvalue; } } ); } } |
測試類:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
|
package net.ydstudio.proxy; import net.ydstudio.service.iuserdao; import net.ydstudio.service.impl.userdao; import org.junit.test; import org.junit.runner.runwith; import org.junit.runners.junit4; import static org.junit. assert .*; /** * @author nick * @projectname javalean * @package net.ydstudio.proxy * @createdate 2018/08/16 15:58 * @updatedate 2018/08/16 15:58 */ @runwith (junit4. class ) public class proxyfactorytest { @test public void test(){ // 目標(biāo)對象 iuserdao target = new userdao(); system.out.println(target.getclass()); // 給目標(biāo)對象,創(chuàng)建代理對象 iuserdao proxy = (iuserdao) new proxyfactory(target).getproxyinstance(); // class $proxy()內(nèi)存中動(dòng)態(tài)生成的代理對象 system.out.println(proxy.getclass()); // 執(zhí)行方法 代理對象 proxy.save(); } } |
jdk實(shí)現(xiàn)代理總結(jié):代理對象不需要實(shí)現(xiàn)接口,但是目標(biāo)對象一定要實(shí)現(xiàn)接口,否則不能用動(dòng)態(tài)代理。
總結(jié)
以上所述是小編給大家介紹的java 中的三種代理模式,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會(huì)及時(shí)回復(fù)大家的。在此也非常感謝大家對服務(wù)器之家網(wǎng)站的支持!
原文鏈接:https://www.ydstudio.net/archives/82.html