什么是代理
代理是設計模式的一種,代理類為委托類提供消息預處理,消息轉發(fā),事后消息處理等功能。Java中的代理分為三種角色:
- 代理類(ProxySubject)
- 委托類(RealSubject)
- 接口(Subject)
三者關系可以表示如下圖:
Java中的代理按照代理類生成時機不同又分為靜態(tài)代理和動態(tài)代理嘀趟。靜態(tài)代理代理類在編譯期就生成,而動態(tài)代理代理類則是在Java運行時動態(tài)生成愈诚。
靜態(tài)代理
Java中的靜態(tài)代理要求代理類(ProxySubject)和委托類(RealSubject)都實現(xiàn)同一個接口(Subject)她按。靜態(tài)代理中代理類在編譯期就已經(jīng)確定,而動態(tài)代理則是JVM運行時動態(tài)生成炕柔,靜態(tài)代理的效率相對動態(tài)代理來說相對高一些酌泰,但是靜態(tài)代理代碼冗余大匕累,一單需要修改接口陵刹,代理類和委托類都需要修改。
舉個例子:
接口(Subject):
interface HelloService {
void sayHello();
}
委托類:
class HelloServiceImpl implements HelloService {
@Override
public void sayHello() {
System.out.println("Hello World!");
}
}
代理類:
class HelloServiceProxy implements HelloService {
private HelloService helloService;
public HelloServiceProxy(HelloService helloService) {
this.helloService = helloService;
}
@Override
public void sayHello() {
System.out.println("Before say hello...");
helloService.sayHello();
System.out.println("After say hello...");
}
}
測試類:
public class HelloServiceProxyTest {
public static void main(String[] args) {
HelloService helloService = new HelloServiceImpl();
HelloServiceProxy proxy = new HelloServiceProxy(helloService);
proxy.sayHello();
}
}
輸出結果:
Before say hello...
Hello World!
After say hello...
動態(tài)代理
Java中的動態(tài)代理依靠反射來實現(xiàn)衰琐,代理類和委托類不需要實現(xiàn)同一個接口。委托類需要實現(xiàn)接口掐隐,否則無法創(chuàng)建動態(tài)代理狗热。代理類在JVM運行時動態(tài)生成,而不是編譯期就能確定虑省。
Java動態(tài)代理主要涉及到兩個類:java.lang.reflect.Proxy
和java.lang.reflect.InvocationHandler
匿刮。代理類需要實現(xiàn)InvocationHandler接口或者創(chuàng)建匿名內(nèi)部類,而Proxy用于創(chuàng)建動態(tài)動態(tài)探颈。
我們用動態(tài)代理來實現(xiàn)HelloService:
接口(Subject):
interface HelloService {
void sayHello();
}
委托類:
class HelloServiceImpl implements HelloService {
@Override
public void sayHello() {
System.out.println("Hello World!");
}
}
動態(tài)代理類:
class HelloServiceDynamicProxy {
private HelloService helloService;
public HelloServiceDynamicProxy(HelloService helloService) {
this.helloService = helloService;
}
public Object getProxyInstance() {
return Proxy.newProxyInstance(helloService.getClass().getClassLoader(), helloService.getClass().getInterfaces(), new InvocationHandler() {
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
System.out.println("Before say hello...");
Object ret = method.invoke(helloService, args);
System.out.println("After say hello...");
return ret;
}
});
}
}
測試類:
public class HelloServieDynamicProxyTest {
public static void main(String[] args){
HelloService helloService = new HelloServiceImpl();
HelloService dynamicProxy = (HelloService) new HelloServiceDynamicProxy(helloService).getProxyInstance();
dynamicProxy.sayHello();
}
}
輸出結果:
Before say hello...
Hello World!
After say hello...
總結
- 靜態(tài)代理實現(xiàn)較簡單熟丸,代理類在編譯期生成,效率高膝擂。缺點是會生成大量的代理類虑啤。
- JDK動態(tài)代理不要求代理類和委托類實現(xiàn)同一個接口,但是委托類需要實現(xiàn)接口架馋,代理類需要實現(xiàn)InvocationHandler接口狞山。
- 動態(tài)代理要求代理類InvocationHandler接口,通過反射代理方法叉寂,比較消耗系統(tǒng)性能萍启,但可以減少代理類的數(shù)量,使用更靈活屏鳍。