你管这叫代理模式(Proxy Pattern)

简介: 代理模式即给一个真实类提供一个代理类,该代理类代替真实类完成其功能,一般还可在代理类上添加一些真实类不具有的附加功能,通俗来讲代理模式就是我们生活中常见的中介,代理模式又可分为静态代理和动态代理。

代理模式


  代理模式即给一个真实类提供一个代理类,该代理类代替真实类完成其功能,一般还可在代理类上添加一些真实类不具有的附加功能,通俗来讲代理模式就是我们生活中常见的中介,代理模式又可分为静态代理动态代理


静态代理


  静态代理需要代理类和目标类实现一样的接口,一般将被代理对象组合到代理类中,然后使用其完成对应功能,并可在代理类中添加额外功能。


租房静态代理过程

第一步:创建服务接口

public interface Rent {
// 出租房屋
void rent();
}


第二步:真实类实现服务接口

public class Host implements Rent{
@Override
public void rent() {
System.out.println("房东要出租房子...");
}
}


第三步:代理类实现服务接口

public class Proxy implements Rent{
// 被代理的类,用于完成被代理类功能。
private Host host;
// 用于初始化被代理类
public void setHost(Host host) {
this.host = host;
}
// 完成被代理类的功能,同时可附加其他功能。
@Override
public void rent() {
doSomething();
host.rent();
doSomething();
}
// 附加的功能
private void doSomething() {
System.out.println("doSomething...");
}
}


第四步:编写测试类

public class Test {
public static void main(String[] args) {
Host host = new Host();
Proxy proxy = new Proxy();
proxy.setHost(host);
proxy.rent();
}
}


静态代理总结


优点

  • 高扩展性:可以在不修改目标类的前提下扩展目标类的功能。

缺点

  • 冗余:一个目标类就会产生一个代理类,会产生过多的代理类。
  • 开发效率低:代理类代码量繁重。
  • 不易维护:一旦接口改变,目标类与代理类都要进行修改。


动态代理


  动态代理中我们不需要手动的创建代理类,只需要编写一个动态处理器来为我们动态创建代理对象,代理对象由 JDK 在运行时动态创建,动态代理基于反射来实现


常用的动态代理方式


  • 基于接口的动态代理:由 JDK 提供,使用 JDK 官方的 Proxy 类创建代理对象。
  • 基于类的动态代理:由第三方 CGLib 提供,使用 CGLib Enhancer 类创建代理对象
  • javassist
  • ……


基于接口的动态代理


  使用基于接口的动态代理首先需要了解 JDK 中的 Proxy 类和 InvocationHandler接口,然后用其编写动态处理器用于完成动态代理。


Proxy


  Proxy 提供了创建动态代理类和实例的静态方法,它也是由这些方法创建的所有动态代理类的超类。
  
一般使用 Proxy newProxyInstance 静态方法创建代理类,需要提供一个类加载器、被代理类的服务接口,以及一个 InvocationHandler 的对象。

  • 类加载器:用于加载类和接口
  • 被代理类的接口:提供被代理类服务接口信息
  • InvocationHandler 对象:用于执行被代理类的方法
public static Object newProxyInstance(ClassLoader loader,
Class<?>[] interfaces,
InvocationHandler h);


InvocationHandler 接口


  当在代理实例上调用方法时,方法调用被编码并分派到其调用处理程序的调用方法,即在 Proxy 中调用方法时,该方法调用被分派给 InvocationHandler 对象中的 invoke 方法执行,由 invoke 方法调用执行被代理对象的方法。

Object invoke(Object proxy, Method method, Object[] args);


动态处理器工具类

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
public class ProxyInvocationHandler implements InvocationHandler {
// 被代理的接口
private Object target;
public ProxyInvocationHandler() {
}
public ProxyInvocationHandler(Object target) {
this.target = target;
}
// 用于初始化被代理类
public void setTarget(Object target) {
this.target = target;
}
// 生成得到代理类
public Object getProxy() {
return Proxy.newProxyInstance(this.getClass().getClassLoader(), target.getClass().getInterfaces(), this);
}
// 处理代理示例并返回结果
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
return method.invoke(target, args);
}
}


基于接口的租房动态代理过程

第一步:创建服务接口

public interface Rent {
// 出租房屋
void rent();
}


第二步:真实类实现服务接口

public class Host implements Rent{
@Override
public void rent() {
System.out.println("房东要出租房子...");
}
}


第三步:编写动态处理器

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
public class ProxyInvocationHandler implements InvocationHandler {
// 被代理的接口
private Object target;
// 用于初始化被代理类
public void setTarget(Object target) {
this.target = target;
}
// 生成得到代理类
public Object getProxy() {
return Proxy.newProxyInstance(this.getClass().getClassLoader(), target.getClass().getInterfaces(), this);
}
// 处理代理示例并返回结果
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
doSomething();
Object invoke = method.invoke(target, args);
doSomething();
return invoke;
}
// 附加的功能
private void doSomething() {
System.out.println("doSomething...");
}
}


第四步:编写测试类

public class Test {
public static void main(String[] args) {
Host host = new Host();
ProxyInvocationHandler proxyInvocationHandler = new ProxyInvocationHandler();
proxyInvocationHandler.setTarget(host);
Rent proxy = (Rent)proxyInvocationHandler.getProxy();
proxy.rent();
}
}


动态代理总结

优点

  • 高扩展性:可以在不修改目标类的前提下扩展目标类的功能。
  • 低冗余:会产生过多的代理类。
  • 低耦合:减少了对业务接口的依赖。
  • 高效开发:大大减少开发任务,一个动态处理器即可完成代理。


注意事项

  • 由于在客户端和真实主题之间增加了代理对象,因此有些类型的代理模式可能会造成请求的处理速度变慢。
  • 实现代理模式需要额外的工作,有些代理模式的实现非常复杂。
  • 和适配器模式的区别:适配器模式主要改变所考虑对象的接口,而代理模式不能改变所代理类的接口。

和装饰器模式的区别:装饰器模式为了增强功能,而代理模式是为了加以控制

相关文章
|
6月前
|
设计模式 Java
设计模式2 - 代理模式【Proxy Pattern】
设计模式2 - 代理模式【Proxy Pattern】
26 0
|
Arthas Java 测试技术
结构型模式 - 代理模式(Proxy Pattern)
结构型模式 - 代理模式(Proxy Pattern)
|
设计模式 安全 Java
从零开始学设计模式(七):代理模式(Proxy Pattern)
代理模式可以说是在日常开发中听到用到最多的设计模式之一了。
204 0
从零开始学设计模式(七):代理模式(Proxy Pattern)
|
设计模式 安全 Java
【Java设计模式】代理模式(Proxy Pattern)
为其他对象提供一种代理,以控制对这个对象的访问,代理对象在客户端和目标对象之间起到中介作用。
112 0
【Java设计模式】代理模式(Proxy Pattern)
|
设计模式 安全 Java
设计模式|从Visitor Pattern说到Pattern Matching
在软件开发领域,我们每次遇到的问题可能都各不相同,有些是跟电商业务相关的,有些是跟底层数据结构相关的,而有些则可能重点在性能优化上。然而不管怎么样,我们在代码层面上解决问题的方法都有一定的共性。有没有人总结过这些共性呢?
设计模式|从Visitor Pattern说到Pattern Matching
|
监控 数据库
代理模式(Proxy)
转载 https://blog.csdn.net/lovelion/article/details/8228042 某软件公司承接了某信息咨询公司的收费商务信息查询系统的开发任务,该系统的基本需求如下: (1) 在进行商务信息查询之前用户需要通过身份验证,只有合法用户才能够使用该查询系统; (2) 在进行商务信息查询时系统需要记录查询日志,以便根据查询次数收取查询费用。
991 0