1. 代理设计模式介绍
代理设计模式,从名字中我们就能知道其大致意思。我们生活中有很多“代理”,比如,租房中介,海外代购等。就是一种本来需要A做的事,让B去代做。
定义:
为其他对象提供一种代理以控制对这个对象的访问。
2. 代理设计模式使用场景
- 当我们无法直接访问某一个对象时,可以通过一个代理对象间接访问。通常委托对象和代理对象有着相同的接口。
- (1)、远程代理,为一个对象在不同的地址空间提供局部代表,这样可以隐藏一个对象存在于不同地址空间的事实。
- (2)、虚拟代理,是根据需要创建开销很大的对象,通过它来存放实例化需要很长时间的真实对象。
- (3)、安全代理,用来控制真实对象访问时的权限。
- (4)、智能引用,当调用真实对象时,代理处理另外一些事。
3. 代理设计模式UML类图
角色介绍:
- Subject: 抽象主题类,该类的主要职责是声明真实主题与代理的共同接口方法,该类既可以是一个抽象类,也可以是一个接口。
- RealSubject:真实主题,该类定义类代理所表示的真实对象,由其执行具体的业务逻辑方法。
- ProxySubject: 代理类,该类持有一个对真实主题的引用,在其所实现的接口方法中调用真实主题对应接口中的方法。
- Client:测试类
4. 代理设计模式简单实现
- (1)、抽象主题类:
public interface Subject {
void visit();
}
抽象主题类中,只有一个visit()方法,代理对象和被代理代理对象均要实现这个方法。
- (2)、真实主题:
public class RealSubject implements Subject {
@Override
public void visit() {
System.out.println("Real Subject");
}
}
真实对象中,继承自抽象主题类,在visit()方法中,实现具体逻辑。
- (3)、代理主题:
public class ProxySubject implements Subject {
private RealSubject mSubject;
public ProxySubject(RealSubject mSubject) {
this.mSubject = mSubject;
}
@Override
public void visit() {
//通过真实主题引用的对象调用真实主题中的逻辑方法
mSubject.visit();
}
}
ProxySubject代理类中有一个真实主题类的引用,同时代理主题类继承了和真实主题类一模一样的抽象类,实现了同样的方法。只不过在代理类中,visit()方法调用了真实主题类的visit()方法。
- (4)、测试类:
public class Client {
public static void main(String[] args) {
//构建一个真实的主题对象
RealSubject realSubject = new RealSubject();
//通过一个真实的主题对象构造一个代理对象
ProxySubject proxySubject = new ProxySubject(realSubject);
//调用代理的相关方法
proxySubject.visit();
}
}
以上模式叫做静态代理,还有一种叫做动态代理模式。
静态代理模式就如上述所示,我们的代码在运行前类的class编译文件就已经存在;而动态代理类则与静态代理类相反,通过反射机制动态地生成代理者对象,代理对象将会在执行阶段执行。
方法如下:我们声明一个动态代理类:
public class DynamicProxy implements InvocationHandler {
private Object object;//被代理对象的引用
public DynamicProxy(Object object) {
this.object = object;
}
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
Object result = method.invoke(object, args);
return result;
}
}
在上述代码中,我们实现了动态代理接口InvocationHandler,这是java为我们提供的,我们只需要重写其调用方法invoke即可。
在以上代码中,我们声明了一个Object的引用,该引用将指向被代理类,而我们在invoke方法中调用具体的被代理方法,也就是真实方法。
接着我们修改Client类方法如下:
public class Client {
public static void main(String[] args) {
RealSubject realSubject = new RealSubject();
//构造一个动态代理对象
DynamicProxy dynamicProxy = new DynamicProxy(realSubject);
//类加载器
ClassLoader classLoader = dynamicProxy.getClass().getClassLoader();
Subject subject = (Subject) Proxy.newProxyInstance(classLoader, new Class[]{Subject.class}, dynamicProxy);
//调用代理对象的visit方法。
subject.visit();
}
}
5. 总结
- 代理设计模式应用比较广泛,缺点较少。
时间: 2024-10-22 00:49:06