场景:
福尔摩斯一直想送礼物给花生,但是羞于直接赠送,于是想到让房东太太去帮忙送礼物.编程如何实现呢?
定义:
为其他对象提供一种代理以控制对这个对象的访问。
角色:
Proxy:代理对象.有下列功能:
实现与具体的目标对象一样的接口,这样就可以使用代理来代替具体的目标对象。
持有一个具体目标对象的引用,可以在需要时调用具体的目标对象。
可以控制对目标对象的访问,并可以负责创建和删除它。
package com.kris.study;
public class Proxy implements Subject {
private RealSubject realSubject = null;
public Proxy(RealSubject realSubject){
this.realSubject = realSubject;
}
@Override
public void request() {
System.out.println("代理之前做一些操作。。。。。。");
realSubject.request();
System.out.println("代理之后做一些操作。。。。。。");
}
}
package com.kris.study;
public interface Subject {
public void request();
}
package com.kris.study;
public class RealSubject implements Subject {
@Override
public void request() {
//执行具体的功能处理
}
}
下面用代理模式实现场景。
package com.kris.study;
public interface SendGift {
public void send();
}
package com.kris.study;
public class Holmes implements SendGift {
@Override
public void send() {
System.out.println("送礼物给花生---来自福尔摩斯");
}
}
package com.kris.study;
public class FangDongTaiTai implements SendGift {
private Holmes holmes = null;
public FangDongTaiTai(Holmes holmes){
this.holmes = holmes;
}
@Override
public void send() {
System.out.println("亲爱的花生,老福让我给你带个东西");
holmes.send();
}
}
package com.kris.study;
public class Client {
public static void main(String[] args) {
Holmes holmes = new Holmes();
SendGift sendGift = new FangDongTaiTai(holmes);
sendGift.send();
}
}
步骤:
1. 创建一个实现接口InvocationHandler的类,它必须实现invoke方法。
2. 创建被代理的类以及接口
3. 通过Proxy的静态方法new ProxyInstance(ClassLoader loader,Class[] interfaces,InvocationHandler h)
创建一个代理,这个代理无论执行什么方法都会转到Handler中的invoke方法去调用.
4. 通过代理调用方法
上面第一步中的invoke方法参数说明
pubilc Object invoke(Object obj,Method method,Object[] args)
obj:代理类
method:被代理的方法
args:方法的参数
动态代理实现场景:
上面的SendGift接口和Holmes类都没变化,现在把房东太太的实现变更为实现InvocationHandler
package com.kris.study;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
public class FangDongTaiTai implements InvocationHandler {
private SendGift holmes = null;
public SendGift bindObject(Holmes holmes){
this.holmes = holmes;
SendGift sendProxy = (SendGift) Proxy.newProxyInstance(holmes.getClass().getClassLoader(),
holmes.getClass().getInterfaces(), this);
return sendProxy;
}
@Override
public Object invoke(Object proxy, Method method, Object[] args)
throws Throwable {
System.out.println("亲爱的花生,这是大福给你的");
return method.invoke(holmes, args);
}
}
package com.kris.study;
public class Client {
public static void main(String[] args) {
Holmes holmes = new Holmes();
FangDongTaiTai giftSendHandler = new FangDongTaiTai();
SendGift proxy = giftSendHandler.bindObject(holmes);
proxy.send();
}
}
代理模式的本质:控制对象的访问
特点:
远程代理:Android中的Binder机制就是通过代理实现,本地通过一个代理对象去调用远程Service的方法。
虚代理:比如加载数据库中所有人员数据,页面上只显示人员名字,点击名字可以查看人员详细信息,这样先用代理对象加载所有人员的名字和ID.当真正需要看其他详细信息时,用代理对象再次去数据库加载所有人员数据。
保护代理:对目标对象的访问进行权限的管控,在代理对象中需要满足某种条件才能调用目标对象的方法。
智能指引:在代理方法的前后执行一些附加的业务逻辑,AOP切面。
何时使用代理:
1.需要为一个对象在不同的地址空间提供局部代表的时候,可以使用远程代理
2.需要创建开销很大的对象时,可以使用虚代理。
3.需要控制对原始对象的访问的时候,可以使用保护代理
4.需要在访问对象执行一些附加操作的时候,可以使用智能指引代理