java代理模式與動(dòng)態(tài)代理模式詳解
1、代理模式
所謂代理,就是一個(gè)人或者一個(gè)機(jī)構(gòu)代表另一個(gè)人或者另一個(gè)機(jī)構(gòu)采取行動(dòng)。在一些情況下,一個(gè)客戶(hù)不想或者不能夠直接引用一個(gè)對(duì)象,而代理對(duì)象可以在客戶(hù)端和目標(biāo)對(duì)象之前起到中介的作用。
代理模式給某一個(gè)對(duì)象提供一個(gè)代理對(duì)象,并由代理對(duì)象控制對(duì)原對(duì)象的引用。
生活中的例子:過(guò)年加班比較忙,沒(méi)空去買(mǎi)火車(chē)票,這時(shí)可以打個(gè)電話到附近的票務(wù)中心,叫他們幫你買(mǎi)張回家的火車(chē)票,當(dāng)然這會(huì)附加額外的勞務(wù)費(fèi)。但要清楚票務(wù)中心自己并不賣(mài)票,只有火車(chē)站才真正賣(mài)票,票務(wù)中心賣(mài)給你的票其實(shí)是通過(guò)火車(chē)站實(shí)現(xiàn)的。這點(diǎn)很重要!
上面這個(gè)例子,你就是“客戶(hù)”,票務(wù)中心就是“代理角色”,火車(chē)站是“真實(shí)角色”,賣(mài)票稱(chēng)為“抽象角色”!
代理模式JAVA代碼示例:
抽象角色:抽象類(lèi)或接口
interface Business
{
void doAction();
}
真實(shí)角色:真正實(shí)現(xiàn)了業(yè)務(wù)邏輯接口
代理角色:自己并未實(shí)現(xiàn)業(yè)務(wù)邏輯接口,而是調(diào)用真實(shí)角色來(lái)實(shí)現(xiàn)
class BusinessImplProxy implements Business
{
private BusinessImpl bi;
public void doAction()
{
if (bi==null)
{
bi = new BusinessImpl();
}
doBefore();
bi.doAction();
doAfter();
}
public void doBefore()
{
System.out.println("前置處理!");
}
public void doAfter()
{
System.out.println("后置處理!");
}
}
//測(cè)試類(lèi)
class Test
{
public static void main(String[] args)
{
//引用變量定義為抽象角色類(lèi)型
Business bi = new BusinessImplProxy();
bi.doAction();
}
}
<span></span>
所以,借助于JVM的支持,可以在運(yùn)行時(shí)動(dòng)態(tài)生成代理類(lèi)(“代理角色”),我們就可以解決上述代理模式中代碼膨脹的問(wèn)題,使用了動(dòng)態(tài)代理后,“代理角色”將不用手動(dòng)生成,而由JVM在運(yùn)行時(shí),通過(guò)指定類(lèi)加載器、接口數(shù)組、調(diào)用處理程序這3個(gè)參數(shù)來(lái)動(dòng)態(tài)生成。
動(dòng)態(tài)代理模式JAVA代碼示例:
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Proxy;
import java.lang.reflect.Method;
//抽象角色:java動(dòng)態(tài)代理的實(shí)現(xiàn)目前只支持接口,不支持抽象類(lèi)
interface BusinessFoo
{
void foo();
}
interface BusinessBar
{
String bar(String message);
}
//真實(shí)角色:真正實(shí)現(xiàn)業(yè)務(wù)邏輯方法
class BusinessFooImpl implements BusinessFoo
{
public void foo()
{
System.out.println("BusinessFooImpl.foo()");
}
}
class BusinessBarImpl implements BusinessBar
{
public String bar(String message)
{
System.out.println("BusinessBarImpl.bar()");
return message;
}
}
//動(dòng)態(tài)角色:動(dòng)態(tài)生成代理類(lèi)
class BusinessImplProxy implements InvocationHandler
{
private Object obj;
BusinessImplProxy() {
}
BusinessImplProxy(Object obj) {
this.obj = obj;
}
public Object invoke(Object proxy,Method method,Object[] args) throws Throwable
{
Object result = null;
doBefore();
result = method.invoke(obj,args);
doAfter();
return result;
}
public void doBefore(){
System.out.println("do something before Business Logic");
}
public void doAfter(){
System.out.println("do something after Business Logic");
}
public static Object factory(Object obj)
{
Class cls = obj.getClass();
return Proxy.newProxyInstance(cls.getClassLoader(),cls.getInterfaces(),new BusinessImplProxy(obj));
}
}
//測(cè)試類(lèi)
public class DynamicProxy
{
public static void main(String[] args) throws Throwable
{
BusinessFooImpl bfoo = new BusinessFooImpl();
BusinessFoo bf = (BusinessFoo)BusinessImplProxy.factory(bfoo);
bf.foo();
System.out.println();
BusinessBarImpl bbar = new BusinessBarImpl();
BusinessBar bb = (BusinessBar)BusinessImplProxy.factory(bbar);
String message = bb.bar("Hello,World");
System.out.println(message);
}
}
程序流程說(shuō)明:
new BusinessFooImpl();創(chuàng)建一個(gè)“真實(shí)角色”,傳遞給工廠方法BusinessImplProxy.factory(),進(jìn)而初始化“調(diào)用處理器”——即實(shí)現(xiàn)InvocationHandler的類(lèi)。并返回一個(gè)動(dòng)態(tài)創(chuàng)建的代理類(lèi)實(shí)例,由于“代理角色”也必然實(shí)現(xiàn)了“抽象角色”提供的業(yè)務(wù)邏輯方法,故可向下轉(zhuǎn)型為BusinessBar,并賦值給指向BusinessBar類(lèi)型的引用bb。
newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h)方法由程序員來(lái)指定參數(shù)動(dòng)態(tài)返回需要的代理類(lèi),而invoke(Object proxy, Method method, Object[] args) 方法則是由JVM在運(yùn)行時(shí)動(dòng)態(tài)調(diào)用的。當(dāng)執(zhí)行“bb.bar("Hello,World");”方法時(shí),JVM動(dòng)態(tài)指派“調(diào)用處理器”,向外層invoke傳遞參數(shù),并調(diào)用method.invoke(obj,args)真正執(zhí)行!
BusinessImplProxy.Factory靜態(tài)方法用來(lái)動(dòng)態(tài)生成代理類(lèi)(“代理角色”),在運(yùn)行時(shí)根據(jù)不同的業(yè)務(wù)邏輯接口BusinessFoo和BusinessBar,在運(yùn)行時(shí)分別動(dòng)態(tài)生成了代理角色。“抽象角色”、“代理角色”以及調(diào)用處理器(實(shí)現(xiàn)InvocationHandler接口的類(lèi))這三者都可以改變,所以說(shuō)JAVA的動(dòng)態(tài)代理十分強(qiáng)大。
相關(guān)文章
springboot獲取真實(shí)ip地址的方法實(shí)例
在使用springboot時(shí),需要獲取訪問(wèn)客戶(hù)端的IP地址,所以下面這篇文章主要給大家介紹了關(guān)于springboot獲取真實(shí)ip地址的相關(guān)資料,需要的朋友可以參考下2022-06-06
mybatis和mybatisplus批量插入問(wèn)題示例詳解
最近在處理一個(gè)功能的時(shí)候,需要批量插入數(shù)據(jù),這篇文章主要給大家介紹了關(guān)于mybatis和mybatisplus批量插入問(wèn)題的相關(guān)資料,文中通過(guò)實(shí)例代碼介紹非常詳細(xì),需要的朋友可以參考下2023-04-04
詳解mybatis-plus的 mapper.xml 路徑配置的坑
這篇文章主要介紹了詳解mybatis-plus的 mapper.xml 路徑配置的坑,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-08-08
Java利用條件運(yùn)算符的嵌套來(lái)完成學(xué)習(xí)成績(jī)的劃分
這篇文章主要介紹了Java利用條件運(yùn)算符的嵌套來(lái)完成學(xué)習(xí)成績(jī)的劃分,需要的朋友可以參考下2017-02-02
SpringBoot整合Freemarker實(shí)現(xiàn)頁(yè)面靜態(tài)化的詳細(xì)步驟
這篇文章主要介紹了SpringBoot整合Freemarker實(shí)現(xiàn)頁(yè)面靜態(tài)化,第一步要?jiǎng)?chuàng)建項(xiàng)目添加依賴(lài),本文分步驟給大家詳細(xì)講解,對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-10-10

