Java通俗易懂系列設(shè)計(jì)模式之代理模式
前言
國(guó)內(nèi)程序員好像普遍對(duì)百度都沒好感,而且百度近些年產(chǎn)生了不少負(fù)面的新聞,像16年的魏則西事件,近期的導(dǎo)演吳京黑白照事件,以及最近作家六六斥百度李彥宏:“你是做搜索引擎還是騙子首領(lǐng)”,還有一件就是與程序員有關(guān)的:搜索Julia語(yǔ)言,在百度和Google得出首條搜索結(jié)果的差異性而被吐槽。Google雖然受歡迎,但是在國(guó)內(nèi)因內(nèi)容審查問題未解決而不能使用,如果我們要使用它就必須使用代理服務(wù)器,由于放置代理服務(wù)器的地區(qū)區(qū)域可以訪問google,所以我們可以先訪問代理服務(wù)器,通過(guò)代理服務(wù)器轉(zhuǎn)發(fā)我們的請(qǐng)求。這是現(xiàn)實(shí)生活中的一種代理模式的實(shí)例,當(dāng)然現(xiàn)實(shí)生活中這種實(shí)例很不少,像明星都有助理,打官司有代理律師等等,這種思想也可以用到我們程序設(shè)計(jì)中。
介紹
在設(shè)計(jì)模式中代理模式可以分為靜態(tài)代理和動(dòng)態(tài)代理,而動(dòng)態(tài)代理根據(jù)代理的對(duì)象類型不同又可以分為Jdk動(dòng)態(tài)代理和Cglib動(dòng)態(tài)代理。
意圖:為其他對(duì)象提供一種代理以控制對(duì)這個(gè)對(duì)象的訪問。
主要解決:在直接訪問對(duì)象時(shí)帶來(lái)的問題,比如說(shuō):要訪問的對(duì)象在遠(yuǎn)程的機(jī)器上。在面向?qū)ο笙到y(tǒng)中,有些對(duì)象由于某些原因(比如對(duì)象創(chuàng)建開銷很大,或者某些操作需要安全控制,或者需要進(jìn)程外的訪問),直接訪問會(huì)給使用者或者系統(tǒng)結(jié)構(gòu)帶來(lái)很多麻煩,我們可以在訪問此對(duì)象時(shí)加上一個(gè)對(duì)此對(duì)象的訪問層。
何時(shí)使用:想在訪問一個(gè)類時(shí)做一些控制。
如何解決:增加中間層。
關(guān)鍵代碼:實(shí)現(xiàn)與被代理類組合。
實(shí)現(xiàn)
近幾年中國(guó)電影行業(yè)蓬勃發(fā)展,電影攝制需要的一種特殊演員->替身,主要任務(wù)是代替影片中原演員表演某些特殊的、高難度的動(dòng)作和技能或原演員所不能勝任的驚險(xiǎn)動(dòng)作,如武打、騎術(shù)、駕車等。拍攝的時(shí)候雖然是替身在拍攝,但是呈現(xiàn)在熒幕前我們觀眾卻不知道是替身而認(rèn)為是明星的真實(shí)拍攝,代理模式也有這種特點(diǎn),雖然是代理類在完成任務(wù),但是呈現(xiàn)出來(lái)的卻是真實(shí)類的實(shí)現(xiàn)。接下來(lái)我們以這種生活中的實(shí)例來(lái)作示例:
公共表演接口的定義
/** 表演 */
public interface Performance {
void act();
}
一.靜態(tài)代理
明星的實(shí)體類
/** 明星 */
public class Actor implements Performance {
@Override
public void act() {
System.out.println("明星上場(chǎng)拍功夫電影");
}
}
替身演員的實(shí)體類
/**
* 替身演員
*/
public class Stuntman implements Performance {
private Actor actor;
@Override
public void act() {
if (actor == null) {
actor = new Actor();
}
System.out.println("替身演員表演跳火車.");
actor.act();
System.out.println("替身演員表演空中360°旋轉(zhuǎn)飛踢.");
}
}
執(zhí)行Demo
public class ProxyPatternDemo {
public static void main(String[] args) {
System.out.println("------電影拍攝開始------");
Performance perform = new Stuntman();
perform.act();
System.out.println("------電影拍攝結(jié)束------");
}
}
執(zhí)行程序,輸出結(jié)果:
------電影拍攝開始------
替身演員表演跳火車.
明星上場(chǎng)拍功夫電影
替身演員表演空中360°旋轉(zhuǎn)飛踢.
二.Jdk動(dòng)態(tài)代理
1、Jdk動(dòng)態(tài)代理是由Java內(nèi)部的反射機(jī)制來(lái)實(shí)現(xiàn)的,目標(biāo)類基于統(tǒng)一的接口InvocationHandler。
2、代理對(duì)象是在程序運(yùn)行時(shí)產(chǎn)生的,而不是編譯期;
3、對(duì)代理對(duì)象的所有接口方法調(diào)用都會(huì)轉(zhuǎn)發(fā)到InvocationHandler.invoke()方法,在invoke()方法里我們可以加入任何邏輯,比如修改方法參數(shù),加入日志功能、安全檢查功能等;之后我們通過(guò)某種方式執(zhí)行真正的方法體,
4、對(duì)于從Object中繼承的方法,JDK動(dòng)態(tài)代理會(huì)把hashCode()、equals()、toString()這三個(gè)非接口方法轉(zhuǎn)發(fā)給InvocationHandler,其余的Object方法則不會(huì)轉(zhuǎn)發(fā)。詳見JDK Proxy官方文檔。
jdk動(dòng)態(tài)代理實(shí)現(xiàn)
public class JdkDynamicProxy implements InvocationHandler {
private Object target;
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
System.out.println("替身演員表演跳火車.");
Object o = method.invoke(target, args);
System.out.println("替身演員表演空中360°旋轉(zhuǎn)飛踢.");
return o;
}
public Object bind(Object target) {
//取得代理對(duì)象
this.target = target;
return Proxy.newProxyInstance(target.getClass().getClassLoader(), target.getClass().getInterfaces(), this);
}
}
執(zhí)行Demo
public static void main(String[] args) {
//創(chuàng)建JDK動(dòng)態(tài)代理類
JdkDynamicProxy proxy = new JdkDynamicProxy();
//綁定對(duì)象
Performance performProxy = (Performance) proxy.bind(new Actor());
System.out.println("------電影拍攝開始------");
performProxy.act();
System.out.println("------電影拍攝結(jié)束------");
}
執(zhí)行結(jié)果
------電影拍攝開始------
替身演員表演跳火車.
明星上場(chǎng)拍功夫電影
替身演員表演空中360°旋轉(zhuǎn)飛踢.
Java動(dòng)態(tài)代理為我們提供了非常靈活的代理機(jī)制,但Jdk動(dòng)態(tài)代理是基于接口的,如果對(duì)象沒有實(shí)現(xiàn)接口我們?cè)撊绾未砟兀看鸢甘荂glib動(dòng)態(tài)代理。
三.Cglib動(dòng)態(tài)代理
cglib動(dòng)態(tài)代理底層則是借助asm來(lái)實(shí)現(xiàn)的,它允許我們?cè)谶\(yùn)行時(shí)對(duì)字節(jié)碼進(jìn)行修改和動(dòng)態(tài)生成,cglib這種第三方類庫(kù)實(shí)現(xiàn)的動(dòng)態(tài)代理應(yīng)用更加廣泛,且在效率上更有優(yōu)勢(shì)。
目標(biāo)類基于統(tǒng)一的接口MethodInterceptor。
CGLIB的核心類:
net.sf.cglib.proxy.Enhancer – 主要的增強(qiáng)類。
net.sf.cglib.proxy.MethodInterceptor – 主要的方法攔截類,它是Callback接口的子接口,需要用戶實(shí)現(xiàn)。
net.sf.cglib.proxy.MethodProxy – JDK的java.lang.reflect.Method類的代理類,可以方便的實(shí)現(xiàn)對(duì)源對(duì)象方法的調(diào)用。
我們要使用cglib代理必須引入cglib的jar包(package net.sf.cglib.proxy;),我在這里使用的是spring包中cglib,其實(shí)和單獨(dú)的引cglib包是一樣的,只不過(guò)spring為了版本不沖突,將cglib包含在自己的包中。
cglib動(dòng)態(tài)代理實(shí)現(xiàn):
import org.springframework.cglib.proxy.Enhancer;
import org.springframework.cglib.proxy.MethodInterceptor;
import org.springframework.cglib.proxy.MethodProxy;
import java.lang.reflect.Method;
public class CglibDynamicProxy implements MethodInterceptor {
private Object target;
//創(chuàng)建代理對(duì)象
public Object getInstance(Object target) {
this.target = target;
Enhancer enhancer = new Enhancer();
enhancer.setSuperclass(this.target.getClass());
// 回調(diào)方法
enhancer.setCallback(this);
// 創(chuàng)建代理對(duì)象
return enhancer.create();
}
@Override
public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
System.out.println("替身演員表演跳火車.");
Object result = methodProxy.invokeSuper(o, objects);
System.out.println("替身演員表演空中360°旋轉(zhuǎn)飛踢.");
return result;
}
}
執(zhí)行Demo
public static void main(String[] args) {
CglibDynamicProxy cglibProxy = new CglibDynamicProxy();
Performance userService = (Performance) cglibProxy.getInstance(new Actor());
System.out.println("------電影拍攝開始------");
userService.act();
System.out.println("------電影拍攝結(jié)束------");
}
執(zhí)行結(jié)果
------電影拍攝開始------
替身演員表演跳火車.
明星上場(chǎng)拍功夫電影
替身演員表演空中360°旋轉(zhuǎn)飛踢.
總結(jié)
1、通過(guò)以上的例子我們可以發(fā)現(xiàn)代理模式的特點(diǎn):
優(yōu)點(diǎn):
- 職責(zé)清晰。
- 高擴(kuò)展性。
- 智能化。
缺點(diǎn):
- 由于在客戶端和真實(shí)主題之間增加了代理對(duì)象,因此有些類型的代理模式可能會(huì)造成請(qǐng)求的處理速度變慢。
- 實(shí)現(xiàn)代理模式需要額外的工作,有些代理模式的實(shí)現(xiàn)非常復(fù)雜。
2、Jdk動(dòng)態(tài)代理和Cglib動(dòng)態(tài)代理的區(qū)別:
- JDK的動(dòng)態(tài)代理機(jī)制只能代理實(shí)現(xiàn)了接口的類,而不能實(shí)現(xiàn)接口的類就不能實(shí)現(xiàn)JDK的動(dòng)態(tài)代理。
- cglib是針對(duì)類來(lái)實(shí)現(xiàn)代理的,他的原理是對(duì)指定的目標(biāo)類生成一個(gè)子類,并覆蓋其中方法實(shí)現(xiàn)增強(qiáng),但因?yàn)椴捎玫氖抢^承,所以不能對(duì)final修飾的類進(jìn)行代理。同樣的,final方法是不能重載的,所以也不能通過(guò)CGLIB代理,遇到這種情況不會(huì)拋異常,而是會(huì)跳過(guò)final方法只代理其他方法。
- JDK動(dòng)態(tài)代理是Java原生支持的,不需要任何外部依賴,但是它只能基于接口進(jìn)行代理;CGLIB通過(guò)繼承的方式進(jìn)行代理,無(wú)論目標(biāo)對(duì)象有沒有實(shí)現(xiàn)接口都可以代理,但是無(wú)法處理final的情況。
- 和適配器模式的區(qū)別:適配器模式主要改變所考慮對(duì)象的接口,而代理模式不能改變所代理類的接口。
- 和裝飾器模式的區(qū)別:裝飾器模式為了增強(qiáng)功能,而代理模式是為了加以控制。
以上就是Java通俗易懂系列設(shè)計(jì)模式之代理模式的詳細(xì)內(nèi)容,更多關(guān)于Java設(shè)計(jì)模式的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Java 中HashCode作用_動(dòng)力節(jié)點(diǎn)Java學(xué)院整理
這篇文章主要介紹了Java 中HashCode作用以及hashcode對(duì)于一個(gè)對(duì)象的重要性,對(duì)java中hashcode的作用相關(guān)知識(shí)感興趣的朋友一起學(xué)習(xí)吧2017-05-05
詳解Java是如何通過(guò)接口來(lái)創(chuàng)建代理并進(jìn)行http請(qǐng)求
今天給大家?guī)?lái)的知識(shí)是關(guān)于Java的,文章圍繞Java是如何通過(guò)接口來(lái)創(chuàng)建代理并進(jìn)行http請(qǐng)求展開,文中有非常詳細(xì)的介紹及代碼示例,需要的朋友可以參考下2021-06-06
在Spring項(xiàng)目中引入高版本依賴并解決低版本沖突問題的解決方法
在Spring項(xiàng)目的開發(fā)過(guò)程中,依賴管理是一個(gè)非常重要且復(fù)雜的問題,我們可能需要引入更高版本的依賴來(lái)使用新特性或修復(fù)舊版本的Bug,然而,這些高版本依賴可能會(huì)與項(xiàng)目中已有的低版本依賴產(chǎn)生沖突,本文將詳細(xì)探討如何在Spring中引入高版本依賴,并解決低版本依賴沖突的問題2025-03-03
基于OpenCv與JVM實(shí)現(xiàn)加載保存圖像功能(JAVA?圖像處理)
openCv有一個(gè)名imread的簡(jiǎn)單函數(shù),用于從文件中讀取圖像,本文給大家介紹JAVA?圖像處理基于OpenCv與JVM實(shí)現(xiàn)加載保存圖像功能,感興趣的朋友一起看看吧2022-01-01
14個(gè)編寫Spring MVC控制器的實(shí)用小技巧(吐血整理)
這篇文章主要介紹了14個(gè)編寫Spring MVC控制器的實(shí)用小技巧(吐血整理),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-11-11
List調(diào)用toString()方法后,去除兩頭的中括號(hào)實(shí)例
下面小編就為大家?guī)?lái)一篇List調(diào)用toString()方法后,去除兩頭的中括號(hào)實(shí)例。希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2017-03-03
Spring Cloud @RefreshScope 原理及使用
這篇文章主要介紹了Spring Cloud @RefreshScope 原理及使用,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-01-01

