当前位置: 首页 > article >正文

Java 代理模式 (Proxy)详解

一、什么是代理模式?

  • 定义: 代理模式是一种结构型设计模式。 它为另一个对象(目标对象/被代理对象)提供一个代理(或占位符),以控制对这个对象的访问。
  • 核心思想: 通过引入一个代理对象,客户端不直接访问目标对象,而是通过代理对象来间接访问目标对象。 代理对象可以控制对目标对象的访问,并可以在访问前后添加额外的操作。
  • 意图: 控制对一个对象的访问,可以延迟加载、访问控制、增强功能等。

二、代理模式的结构

代理模式通常包含以下几个角色:

  1. Subject (抽象主题):

    • 定义了 RealSubject 和 Proxy 的共同接口。 客户端通过 Subject 接口访问目标对象。
    • 通常是一个接口或抽象类。
  2. RealSubject (真实主题/目标对象/被代理对象):

    • 定义了真正的业务逻辑。
    • 实现了 Subject 接口。
  3. Proxy (代理):

    • 持有 RealSubject 对象的引用。
    • 实现了 Subject 接口,与 RealSubject 具有相同的方法。
    • 控制对 RealSubject 对象的访问,并可以在访问前后添加额外的操作。
    • 客户端通过 Proxy 对象间接访问 RealSubject 对象。

UML 类图:

+----------------+       +----------------+       +----------------+
|   <<Subject>>   |       |     Proxy      |       |  RealSubject   |
+----------------+       +----------------+       +----------------+
| +request()     |------>| -realSubject   |------>| +request()     |
+----------------+       | +request()     |       +----------------+
                             +preRequest()
                             +postRequest()

三、代理模式的类型

根据代理的创建时间和功能,可以将代理模式分为以下几种类型:

  1. 静态代理 (Static Proxy):

    • 特点: 在编译时就已经确定了代理类和被代理类之间的关系。 代理类和被代理类都需要实现相同的接口。
    • 优点: 实现简单,易于理解。
    • 缺点:
      • 代码冗余: 如果需要代理多个类,就需要创建多个代理类,导致代码冗余。
      • 可维护性差: 如果接口发生变化,代理类和被代理类都需要进行修改。
  2. 动态代理 (Dynamic Proxy):

    • 特点: 在运行时动态地生成代理类,无需手动创建代理类。
    • 优点:
      • 灵活性高: 可以代理任何实现了接口的类,无需修改原始代码。
      • 代码复用: 可以使用同一个代理类来代理多个不同的类。
      • 可维护性好: 如果接口发生变化,只需要修改代理逻辑,无需修改被代理类。
    • 缺点:
      • 实现复杂: 动态代理的实现比静态代理复杂。
      • 性能开销: 动态代理需要使用反射机制,性能比静态代理略低。
    • Java 中的动态代理:
      • JDK 动态代理: Java 提供的内置动态代理机制,只能代理实现了接口的类。
      • CGLIB 动态代理: 第三方库提供的动态代理机制,可以代理没有实现接口的类。
  3. 其他代理

    • 保护代理 用于控制对敏感对象的访问。
    • 远程代理 用于访问远程对象。
    • 虚拟代理 通过代理延迟创建开销大的对象

四、代理模式的实现方式 (Java)

  1. 静态代理:

    // 抽象主题
    interface Image {
        void display();
    }
    
    // 真实主题
    class RealImage implements Image {
        private String filename;
    
        public RealImage(String filename) {
            this.filename = filename;
            loadImageFromDisk();
        }
    
        private void loadImageFromDisk() {
            System.out.println("Loading image: " + filename);
        }
    
        @Override
        public void display() {
            System.out.println("Displaying image: " + filename);
        }
    }
    
    // 代理
    class ImageProxy implements Image {
        private RealImage realImage;
        private String filename;
    
        public ImageProxy(String filename) {
            this.filename = filename;
        }
    
        @Override
        public void display() {
            if (realImage == null) {
                realImage = new RealImage(filename);
            }
            realImage.display();
        }
    }
    
    // 客户端代码
    public class StaticProxyExample {
        public static void main(String[] args) {
            Image image = new ImageProxy("test_image.jpg");
    
            // 第一次调用 display() 方法,会加载图片
            image.display();
    
            // 第二次调用 display() 方法,不会再次加载图片
            image.display();
        }
    }
    
  2. JDK 动态代理:

    import java.lang.reflect.InvocationHandler;
    import java.lang.reflect.Method;
    import java.lang.reflect.Proxy;
    
    // 抽象主题
    interface Image {
        void display();
    }
    
    // 真实主题
    class RealImage implements Image {
        private String filename;
    
        public RealImage(String filename) {
            this.filename = filename;
            loadImageFromDisk();
        }
    
        private void loadImageFromDisk() {
            System.out.println("Loading image: " + filename);
        }
    
        @Override
        public void display() {
            System.out.println("Displaying image: " + filename);
        }
    }
    
    // 调用处理器
    class ImageInvocationHandler implements InvocationHandler {
        private Object target; // 被代理的对象
    
        public ImageInvocationHandler(Object target) {
            this.target = target;
        }
    
        @Override
        public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
            // 在调用目标方法之前执行的操作
            System.out.println("Before invoking method: " + method.getName());
    
            // 调用目标方法
            Object result = method.invoke(target, args);
    
            // 在调用目标方法之后执行的操作
            System.out.println("After invoking method: " + method.getName());
    
            return result;
        }
    }
    
    // 客户端代码
    public class JDKDynamicProxyExample {
        public static void main(String[] args) {
            // 创建被代理对象
            Image realImage = new RealImage("test_image.jpg");
    
            // 创建调用处理器
            ImageInvocationHandler handler = new ImageInvocationHandler(realImage);
    
            // 创建代理对象
            Image proxy = (Image) Proxy.newProxyInstance(
                    Image.class.getClassLoader(), // 类加载器
                    new Class[] {Image.class}, // 代理类实现的接口列表
                    handler // 调用处理器
            );
    
            // 通过代理对象调用方法
            proxy.display();
        }
    }
    
  3. CGLIB 动态代理:

    import net.sf.cglib.proxy.Enhancer;
    import net.sf.cglib.proxy.MethodInterceptor;
    import net.sf.cglib.proxy.MethodProxy;
    
    import java.lang.reflect.Method;
    
    // 真实主题 (不需要实现接口)
    class RealImage {
        private String filename;
    
        public RealImage(String filename) {
            this.filename = filename;
            loadImageFromDisk();
        }
        public RealImage(){} //需要一个无参构造函数
    
        private void loadImageFromDisk() {
            System.out.println("Loading image: " + filename);
        }
    
        public void display() {
            System.out.println("Displaying image: " + filename);
        }
    }
    
    // 方法拦截器
    class ImageMethodInterceptor implements MethodInterceptor {
        @Override
        public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {
            // 在调用目标方法之前执行的操作
            System.out.println("Before invoking method: " + method.getName());
    
            // 调用目标方法
            Object result = proxy.invokeSuper(obj, args);
    
            // 在调用目标方法之后执行的操作
            System.out.println("After invoking method: " + method.getName());
    
            return result;
        }
    }
    
    // 客户端代码
    public class CGLIBDynamicProxyExample {
        public static void main(String[] args) {
            // 创建 Enhancer 对象
            Enhancer enhancer = new Enhancer();
    
            // 设置超类
            enhancer.setSuperclass(RealImage.class);
    
            // 设置回调
            enhancer.setCallback(new ImageMethodInterceptor());
    
            // 创建代理对象
            RealImage proxy = (RealImage) enhancer.create();
    
            // 通过代理对象调用方法
            proxy.display();
        }
    }
    

五、代理模式的优缺点

优点:

  • 职责清晰: 将客户端与目标对象分离,降低了耦合度。
  • 扩展性好: 可以通过添加新的代理类来扩展系统功能,而无需修改原始代码。
  • 保护目标对象: 代理对象可以控制对目标对象的访问,保护目标对象免受恶意访问。
  • 增强目标对象的功能: 代理对象可以在访问目标对象前后添加额外的操作,例如日志记录、安全检查、延迟加载等。

缺点:

  • 增加系统复杂性: 引入代理对象会增加系统的复杂性。
  • 可能降低性能: 代理对象需要进行额外的处理,可能会降低程序的性能。

六、代理模式的应用场景

  • 远程代理 (Remote Proxy): 为远程对象提供一个本地代理,隐藏远程对象的具体实现细节。
  • 虚拟代理 (Virtual Proxy): 为创建开销大的对象提供一个代理,延迟对象的创建,直到真正需要使用时才创建。
  • 保护代理 (Protection Proxy): 控制对敏感对象的访问,只允许具有特定权限的客户端访问。
  • 缓存代理 (Cache Proxy): 为访问开销大的对象提供一个缓存,提高访问效率。
  • 智能引用代理 (Smart Reference Proxy): 在访问对象时执行额外的操作,例如引用计数、对象锁定等。
  • AOP (面向切面编程): 使用动态代理来实现 AOP,例如 Spring AOP。
  • 延迟加载 (Lazy Loading): 例如,Hibernate 中的延迟加载机制。
  • 防火墙代理: 控制网络访问,保护内部网络。

七、总结

代理模式是一种非常有用的设计模式,它可以控制对对象的访问,并可以在访问前后添加额外的操作。 在 Java 中,可以使用静态代理、JDK 动态代理和 CGLIB 动态代理来实现代理模式。 选择哪种实现方式取决于具体的应用场景和需求。


http://www.kler.cn/a/560579.html

相关文章:

  • 最小化重投影误差求解PnP
  • 【ICE】飞冰项目中根据不同域名设置不同的网站logo
  • Python Django系列—入门实例(二)
  • 全面解析 Node-RED:功能、Docker 部署与实战示例
  • 伪404兼容huawei生效显示404
  • windows server 2016 安装 sqlserver2016数据库
  • GCC编译器(含预处理/编译/汇编/链接四阶段详解)
  • Visual Studio 安装全攻略
  • flutter Column嵌套ListView高度自适应问题
  • MySQL的InnoDB引擎中的聚簇索引和非聚簇索引有什么区别?
  • 如何用HBase轻松管理海量数据?
  • 蓝桥杯备赛-迷宫-BFS
  • leetcode_二叉树 543.二叉树的直径
  • GreatSQL修改配置文件参数无法生效
  • jvm调试和查看工具
  • 与go比肩的FastAPI,如何快速入门
  • Java 大视界 -- 深入剖析 Java 大数据实时 ETL 中的数据质量保障策略(97)
  • go实现敏感词过滤
  • 我与Linux的爱恋:了解信号量+共享内存+消息队列的应用
  • 【quicker】调节PPT指定字号字体大小/快速调节WPS的PPT字体大小