日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程语言 > java >内容正文

java

Java设计模式之代理模式(Proxy)

發布時間:2025/3/21 java 29 豆豆
生活随笔 收集整理的這篇文章主要介紹了 Java设计模式之代理模式(Proxy) 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

參考http://blog.csdn.net/jianghuxiaoxiami/article/details/3403924


1.代理模式

?

代理模式的作用是:為其他對象提供一種代理以控制對這個對象的訪問。在某些情況下,一個客戶不想或者不能直接引用另一個對象,而代理對象可以在客戶端和目標對象之間起到中介的作用。

?

代理模式一般涉及到的角色有:

?

抽象角色:聲明真實對象和代理對象的共同接口;

?

代理角色:代理對象角色內部含有對真實對象的引用,從而可以操作真實對象,同時代理對象提供與真實對象相同的接口以便在任何時刻都能代替真實對象。同時,代理對象可以在執行真實對象操作時,附加其他的操作,相當于對真實對象進行封裝。

?

真實角色:代理角色所代表的真實對象,是我們最終要引用的對象。

?

?

以下是《Java與模式》中的示例為例:

?

1.1 靜態代理



[java]?view plaincopy
  • package?staticproxy;??
  • ??
  • /**?
  • ?*?抽象角色?
  • ?*/??
  • public?abstract?class?Subject?{??
  • ????public?abstract?void?request();??
  • }??



  • [java]?view plaincopy
  • package?staticproxy;??
  • ??
  • /**?
  • ?*?真實的角色?
  • ?*/??
  • public?class?RealSubject?extends?Subject?{??
  • ??
  • ????@Override??
  • ????public?void?request()?{??
  • ????????//?TODO?Auto-generated?method?stub??
  • ??
  • ????}??
  • ??
  • }??



  • [java]?view plaincopy
  • package?staticproxy;??
  • ??
  • /**?
  • ?*?靜態代理,對具體真實對象直接引用?
  • ?*?代理角色,代理角色需要有對真實角色的引用,?
  • ?*?代理做真實角色想做的事情?
  • ?*/??
  • public?class?ProxySubject?extends?Subject?{??
  • ??????
  • ????private?RealSubject?realSubject?=?null;??
  • ??????
  • ????/**?
  • ?????*?除了代理真實角色做該做的事情,代理角色也可以提供附加操作,?
  • ?????*?如:preRequest()和postRequest()?
  • ?????*/??
  • ????@Override??
  • ????public?void?request()?{??
  • ????????preRequest();??//真實角色操作前的附加操作??
  • ??????????
  • ????????if(realSubject?==?null){??
  • ????????????realSubject?=??new?RealSubject();??
  • ????????}??
  • ????????realSubject.request();??
  • ??????????
  • ????????postRequest();??//真實角色操作后的附加操作??
  • ????}??
  • ??
  • ????/**?
  • ?????*??真實角色操作前的附加操作?
  • ?????*/??
  • ????private?void?postRequest()?{??
  • ????????//?TODO?Auto-generated?method?stub??
  • ??????????
  • ????}??
  • ??
  • ????/**?
  • ?????*??真實角色操作后的附加操作?
  • ?????*/??
  • ????private?void?preRequest()?{??
  • ????????//?TODO?Auto-generated?method?stub??
  • ??????????
  • ????}??
  • ??
  • }??



  • [java]?view plaincopy
  • package?staticproxy;??
  • ??
  • /**?
  • ?*??客戶端調用??
  • ?*/??
  • public?class?Main?{??
  • ????public?static?void?main(String[]?args)?{??
  • ????????Subject?subject?=?new?ProxySubject();??
  • ????????subject.request();??//代理者代替真實者做事情??
  • ????}??
  • }??




  • 以賣房(Subject的request方法)為例,賣房者(RealSubject)想賣房,但是不想搞那么多麻煩事,不想天天招買房的騷擾,就委托中介(ProxySubject)去幫忙賣房,中介幫忙賣房,當然不會簡單的賣房,可能還需要收取中介費(preRequest()和postRequest()方法)等等附加收費,最終買房者(客戶端Main)買房,找到的是中介,不會與房主接觸,與房主接觸的是中介,買房者最終跟中介買房,完成操作。

    ?

    ?

    這個是靜態代理。真實角色必須是事先已經存在的,并將其作為代理對象的內部屬性。但是實際使用時,一個真實角色必須對應一個代理角色,如果大量使用會導致類的急劇膨脹;此外,如果事先并不知道真實角色,該如何使用代理呢?這個問題可以通過Java的動態代理類來解決。

    ?

    動態代理,就相當于代理者不僅僅只是代理一個真實對象,也可以代理很多對象,而且對象是動態指定的。

    ?

    1.2 動態代理

    ?

    Java動態代理類位于Java.lang.reflect包下,一般主要涉及到以下兩個類:

    ?

    (1). Interface InvocationHandler:該接口中僅定義了一個方法Object:invoke(Objectobj,Method method, Object[] args)。在實際使用時,第一個參數obj一般是指代理類,method是被代理的方法,如上例中的request(),args為該方法的參數數組。這個抽象方法在代理類中動態實現。

    ?

    (2).Proxy:該類即為動態代理類,作用類似于上例中的ProxySubject,其中主要包含以下內容:

    Protected Proxy(InvocationHandler h):構造函數,估計用于給內部的h賦值。

    ?

    Static Class getProxyClass (ClassLoaderloader, Class[] interfaces):獲得一個代理類,其中loader是類裝載器,interfaces是真實類所擁有的全部接口的數組。

    ?

    Static Object newProxyInstance(ClassLoaderloader, Class[] interfaces, InvocationHandler h):返回代理類的一個實例,返回后的代理類可以當作被代理類使用(可使用被代理類的在Subject接口中聲明過的方法)。

    ?

    所謂Dynamic Proxy是這樣一種class:它是在運行時生成的class,在生成它時你必須提供一組interface給它,然后該class就宣稱它實現了這些interface。你當然可以把該class的實例當作這些interface中的任何一個來用。當然啦,這個DynamicProxy其實就是一個Proxy,它不會替你作實質性的工作,在生成它的實例時你必須提供一個handler,由它接管實際的工作。(參見文獻3)

    ?

    在使用動態代理類時,我們必須實現InvocationHandler接口

    ?

    [java]?view plaincopy
  • package?dynamicproxy;??
  • ??
  • /**?
  • ?*?抽象角色?
  • ?*?這里應改為接口?
  • ?*/??
  • public?interface?Subject?{??
  • ????void?request();??
  • }??


  • [java]?view plaincopy
  • package?dynamicproxy;??
  • ??
  • /**?
  • ?*?真實的角色?
  • ?*?實現接口?
  • ?*/??
  • public?class?RealSubject?implements?Subject?{??
  • ??
  • ????@Override??
  • ????public?void?request()?{??
  • ????????//?TODO?Auto-generated?method?stub??
  • ??
  • ????}??
  • ??
  • }??



  • [java]?view plaincopy
  • package?dynamicproxy;??
  • ??
  • import?java.lang.reflect.InvocationHandler;??
  • import?java.lang.reflect.Method;??
  • ??
  • /**?
  • ?*?動態代理,?它是在運行時生成的class,在生成它時你必須提供一組interface給它,?然后該class就宣稱它實現了這些interface。?
  • ?*?你當然可以把該class的實例當作這些interface中的任何一個來用。?當然啦,這個Dynamic?
  • ?*?Proxy其實就是一個Proxy,它不會替你作實質性的工作,?在生成它的實例時你必須提供一個handler,由它接管實際的工作。?
  • ?*/??
  • public?class?DynamicSubject?implements?InvocationHandler?{??
  • ??
  • ????private?Object?sub;?//?真實對象的引用??
  • ??
  • ????public?DynamicSubject(Object?sub)?{??
  • ????????this.sub?=?sub;??
  • ????}??
  • ??
  • ????@Override??
  • ????public?Object?invoke(Object?proxy,?Method?method,?Object[]?args)?throws?Throwable?{??
  • ????????System.out.println("before?calling?"?+?method);???
  • ????????method.invoke(sub,args);???
  • ????????System.out.println("after?calling?"?+?method);???
  • ????????return?null;???
  • ????}??
  • ??
  • }??




  • [java]?view plaincopy
  • package?dynamicproxy;??
  • ??
  • import?java.lang.reflect.Constructor;??
  • import?java.lang.reflect.InvocationHandler;??
  • import?java.lang.reflect.Proxy;??
  • ??
  • public?class?Main?{??
  • ????public?static?void?main(String[]?args)?throws?Throwable?{??
  • ????????RealSubject?rs?=?new?RealSubject();??
  • ????????InvocationHandler?handler?=?new?DynamicSubject(rs);??
  • ????????Class?cls?=?rs.getClass();??
  • ????????//以下是分解步驟??
  • ????????/*?
  • ????????Class?c?=?Proxy.getProxyClass(cls.getClassLoader(),?cls.getInterfaces());?
  • ????????Constructor?ct?=?c.getConstructor(new?Class[]{InvocationHandler.class});?
  • ????????Subject?subject?=(Subject)?ct.newInstance(new?Object[]{handler});?
  • ????????*/??
  • ??????????
  • ????????//以下是一次性生成??
  • ????????Subject?subject?=?(Subject)Proxy.newProxyInstance(cls.getClassLoader(),cls.getInterfaces(),?handler);??
  • ????????subject.request();??
  • ????}??
  • }??


  • 通過這種方式,被代理的對象(RealSubject)可以在運行時動態改變,需要控制的接口(Subject接口)可以在運行時改變,控制的方式(DynamicSubject類)也可以動態改變,從而實現了非常靈活的動態代理關系

    ?

    1.3.代理模式使用原因和應用方面

    ?

    (1)授權機制不同級別的用戶對同一對象擁有不同的訪問權利,如Jive論壇系統中,就使用Proxy進行授權機制控制,訪問論壇有兩種人:注冊用戶和游客(未注冊用戶),Jive中就通過類似ForumProxy這樣的代理來控制這兩種用戶對論壇的訪問權限.

    ?

    (2)某個客戶端不能直接操作到某個對象,但又必須和那個對象有所互動.

    ???? 舉例兩個具體情況:

    ???? 如果那個對象是一個是很大的圖片,需要花費很長時間才能顯示出來,那么當這個圖片包含在文檔中時,使用編輯器或瀏覽器打開這個文檔,打開文檔必須很迅速,不能等待大圖片處理完成,這時需要做個圖片Proxy來代替真正的圖片.

    ????

    ???? 如果那個對象在Internet的某個遠端服務器上,直接操作這個對象因為網絡速度原因可能比較慢,那我們可以先用Proxy來代替那個對象.

    ?

    ???? 總之原則是,對于開銷很大的對象,只有在使用它時才創建,這個原則可以為我們節省很多寶貴的Java內存. 所以,有些人認為Java耗費資源內存,我以為這和程序編制思路也有一定的關系.

    ?

    (3)現實中,Proxy應用范圍很廣,現在流行的分布計算方式RMI和Corba等都是Proxy模式的應用


    from:?https://blog.csdn.net/liangbinny/article/details/18656791

    總結

    以上是生活随笔為你收集整理的Java设计模式之代理模式(Proxy)的全部內容,希望文章能夠幫你解決所遇到的問題。

    如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。