更新時間:2022-06-07 10:12:47 來源:動力節(jié)點 瀏覽848次
1.代理模式
(1)目的:代理模式的目的是為其他對象提供一個代理來控制對該對象的訪問。使用代理包裝對象并將原始對象替換為代理對象。對原始對象的任何調(diào)用都是通過代理進行的。代理對象決定是否以及何時將方法調(diào)用轉(zhuǎn)移到原始對象。
(2)主要解決方案:直接訪問對象時出現(xiàn)的問題,例如:要訪問的對象在遠程機器上。在面向?qū)ο蟮南到y(tǒng)中,有些對象由于某種原因(例如,對象創(chuàng)建成本高,某些操作需要安全控制,或者進程外訪問),直接訪問會給用戶或系統(tǒng)結(jié)構(gòu)帶來很多麻煩,當我們訪問它時,我們可以為這個對象添加一個訪問層。
(3)何時使用:你想在訪問一個類時做一些控制。
(4)優(yōu)勢:職責明確,擴展性高,智能化。
(5)缺點:由于在客戶端和真實主體之間增加了代理對象,某些類型的代理模式可能會導致請求處理更慢;實現(xiàn)代理模式需要額外的工作,并且一些代理模式實現(xiàn)起來非常復雜。
2.靜態(tài)代理
interface TicketOffice{
void sellTicket();
}
// Proxy Class
class RailwayStation implements TicketOffice{
@Override
public void sellTicket() {
System.out.println("The station really sold a ticket");
}
}
// proxy class
class ProxyStation implements TicketOffice{
private TicketOffice ticketOffice;
public ProxyStation(TicketOffice ticketOffice){
this.ticketOffice = ticketOffice;
}
@Override
public void sellTicket() {
System.out.println("Work before ticket sales");
ticketOffice.sellTicket();
System.out.println("Work after ticket sale");
}
}
// Test Class
public class StaticProxyTest {
public static void main(String[] args) {
// Create an object of a proxy class
TicketOffice rs = new RailwayStation();
// Object to create proxy class
TicketOffice ps = new ProxyStation(rs);
ps.sellTicket();
}
}
3.動態(tài)代理
執(zhí)行靜態(tài)代理時,代理類和目標對象的類是在編譯時確定的,不利于程序的擴展。同時,每個代理類只能服務(wù)一個接口,這樣在程序的開發(fā)中難免會出現(xiàn)過多的代理。最好通過一個代理類來完成所有代理功能。
使用反射機制在運行時創(chuàng)建代理類的動態(tài)代理。
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
interface Hello {
void sayHello();
}
// Proxy Class
class HelloImpl implements Hello{
@Override
public void sayHello() {
System.out.println("Hello world!");
}
}
class MyInvocationHandler2 implements InvocationHandler {
private Object target; // Target object
public MyInvocationHandler2(Object target){
this.target = target;
}
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
System.out.println("Pre-notification code");
// Execute the appropriate target method
Object rs = method.invoke(target, args);
System.out.println("Post-processing code");
return rs;
}
}
public class ProxyTest2 {
public static void main(String[] args) {
HelloImpl helloImpl = new HelloImpl();
Hello hello = (Hello) Proxy.newProxyInstance(Hello.class.getClassLoader(), new Class[]{Hello.class}, new MyInvocationHandler2(helloImpl));
hello.sayHello();
}
}