1、簡述 代理模式中可以映射為現實生活中的生產者、中介商、消費者,生產者可抽象為委托類,中介商可抽象為代理類,消費者可以抽象為調用者對象。代理模式可以簡化消費者購買商品的模式,比如超市裡面可以購買各種商品,消費者只需要找到超市和超市裡面商品即可,不需跑到各個生產商品的工廠。 優點一:可以隱藏委托類的 ...
1、簡述
代理模式中可以映射為現實生活中的生產者、中介商、消費者,生產者可抽象為委托類,中介商可抽象為代理類,消費者可以抽象為調用者對象。代理模式可以簡化消費者購買商品的模式,比如超市裡面可以購買各種商品,消費者只需要找到超市和超市裡面商品即可,不需跑到各個生產商品的工廠。
優點一:可以隱藏委托類的實現;
優點二:可以實現客戶與委托類間的解耦,在不修改委托類代碼的情況下能夠做一些額外的處理。
2、靜態代理類
若代理類在程式運行前就已經存在,那麼這種代理方式被成為 靜態代理 ,這種情況下的代理類通常都是我們在Java代碼中定義的。 通常情況下, 靜態代理中的代理類和委托類會實現同一介面或是派生自相同的父類。
3、動態代理
(1)InvocationHandler介面
在使用動態代理時,我們需要定義一個位於代理類與委托類之間的中介類,這個中介類被要求實現InvocationHandler介面,這個介面的定義如下:
public
interface
InvocationHandler {
Object invoke(Object proxy, Method method, Object[] args);
}
(2)委托類的定義
動態代理方式下,要求委托類必須實現某個介面,這裡我們實現的是Sell介面。委托類Vendor類的定義如下:
public
class
Vendor
implements
Sell {
public
void
sell() {
System.out.println(
"In sell method"
);
}
public
void
ad() {
System,out.println(
"ad method"
)
}
}
(3)中介類
上面我們提到過,中介類必須實現InvocationHandler介面,作為調用處理器”攔截“對代理類方法的調用。中介類的定義如下:
public
class
DynamicProxy
implements
InvocationHandler {
private
Object obj;
//obj為委托類對象;
public
DynamicProxy(Object obj) {
this
.obj = obj;
}
@Override
public
Object invoke(Object proxy, Method method, Object[] args)
throws
Throwable {
System.out.println(
"before"
);
Object result = method.invoke(obj, args);
System.out.println(
"after"
);
return
result;
}
}
(4)動態生成代理類
public
class
Main {
public
static
void
main(String[] args) {
//創建中介類實例
DynamicProxy inter =
new
DynamicProxy(
new
Vendor());
//加上這句將會產生一個$Proxy0.class文件,這個文件即為動態生成的代理類文件
System.getProperties().put(
"sun.misc.ProxyGenerator.saveGeneratedFiles"
,
"true"
);
//獲取代理類實例sell
Sell sell = (Sell)(Proxy.newProxyInstance(Sell.
class
.getClassLoader(),
new
Class[] {Sell.
class
}, inter));
//通過代理類對象調用代理類方法,實際上會轉到invoke方法調用
sell.sell();
sell.ad();
}
}
(5)運行結果
before
In sell method
after
before
In ad method
after