Java中的代理模式

在Java中,代理模式用于为对象提供一种间接访问方式,常用于增强功能(如日志、事务等)。代理分为静态代理动态代理,两者实现方式和原理不同。


一、静态代理

实现过程

  1. 定义公共接口

    代理类和被代理类需实现同一接口。

    java

    csharp 复制代码
    public interface UserService {
        void save();
    }
  2. 实现被代理类(真实对象)

    java

    typescript 复制代码
    public class UserServiceImpl implements UserService {
        @Override
        public void save() {
            System.out.println("保存用户数据");
        }
    }
  3. 实现代理类

    代理类持有被代理对象的引用,并在方法调用前后添加增强逻辑。

    java

    csharp 复制代码
    public class UserServiceProxy implements UserService {
        private final UserService target;  // 持有真实对象
    
        public UserServiceProxy(UserService target) {
            this.target = target;
        }
    
        @Override
        public void save() {
            System.out.println("前置增强(如日志)");
            target.save();  // 调用真实对象的方法
            System.out.println("后置增强(如事务提交)");
        }
    }
  4. 使用代理

    java

    java 复制代码
    public static void main(String[] args) {
        UserService realService = new UserServiceImpl();
        UserService proxy = new UserServiceProxy(realService);
        proxy.save();  // 通过代理调用方法
    }

原理

  • 编译时绑定:代理类在编译期已确定,需手动编写代码。
  • 直接调用:代理类通过组合持有真实对象,显式调用其方法。
  • 缺点:每个被代理类需单独编写代理类,代码冗余。

二、动态代理

实现过程(基于JDK内置Proxy类)

  1. 定义接口(同上)

    java

    csharp 复制代码
    public interface UserService {
        void save();
    }
  2. 实现被代理类(同上)

    java

    java 复制代码
    public class UserServiceImpl implements UserService { ... }
  3. 实现InvocationHandler

    定义代理逻辑的通用处理器。

    java

    typescript 复制代码
    public class LogHandler implements InvocationHandler {
        private final Object target;  // 持有任意真实对象
    
        public LogHandler(Object target) {
            this.target = target;
        }
    
        @Override
        public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
            System.out.println("前置增强:" + method.getName());
            Object result = method.invoke(target, args);  // 反射调用真实对象方法
            System.out.println("后置增强");
            return result;
        }
    }
  4. 动态生成代理对象

    使用Proxy.newProxyInstance创建代理实例。

    java

    scss 复制代码
    public static void main(String[] args) {
        UserService realService = new UserServiceImpl();
        
        // 动态生成代理对象
        UserService proxy = (UserService) Proxy.newProxyInstance(
            realService.getClass().getClassLoader(),  // 类加载器
            realService.getClass().getInterfaces(),   // 实现的接口
            new LogHandler(realService)               // InvocationHandler
        );
    
        proxy.save();  // 调用代理方法
    }

原理

  1. 运行时动态生成代理类

    • JDK的Proxy类在运行时生成代理类的字节码(类名通常为$Proxy0)。
    • 通过sun.misc.ProxyGenerator生成字节码,并加载到JVM。
  2. 代理类结构

    生成的代理类继承Proxy并实现目标接口:

    java

    scala 复制代码
    public final class $Proxy0 extends Proxy implements UserService {
        public $Proxy0(InvocationHandler h) {
            super(h);
        }
    
        @Override
        public void save() {
            super.h.invoke(this,  // 调用InvocationHandler的invoke方法
                UserService.class.getMethod("save"), 
                null);
        }
    }

关键点

  • JDK动态代理限制 :只能代理接口 ,不能代理类(因Java单继承,代理类已继承Proxy)。
  • 性能开销:反射调用方法比直接调用略慢(但现代JVM已优化)。
  • 替代方案 :若需代理类而非接口,可用CGLIB(通过继承实现代理)。

三、对比总结

特性 静态代理 动态代理(JDK)
实现时机 编译期(手动编写代理类) 运行期(自动生成代理类)
代码冗余 每个被代理类需单独编写代理类 通用(一个处理器处理多个类)
灵活性 低(修改需重新编译) 高(动态配置)
代理对象类型 类或接口 仅接口
性能 无反射开销(直接调用) 有反射开销(但可接受)
典型应用 简单场景、少量代理 Spring AOP、RPC框架等

四、动态代理底层原理(字节码层面)

  1. 生成代理类字节码
    Proxy.newProxyInstance()调用ProxyGenerator.generateProxyClass()生成字节码(可通过-Dsun.misc.ProxyGenerator.saveGeneratedFiles=true导出.class文件)。
  2. 加载代理类
    通过ClassLoader将字节码加载到JVM。
  3. 实例化代理对象
    利用反射创建代理实例,构造器传入InvocationHandler
  4. 方法调用转发
    代理类的方法调用全部委托给InvocationHandler.invoke(),由它决定增强逻辑和真实对象的调用。

五、扩展:CGLIB动态代理

若需代理(非接口),可用CGLIB库:

java

typescript 复制代码
// 1. 实现MethodInterceptor
public class CglibProxy implements MethodInterceptor {
    @Override
    public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {
        System.out.println("前置增强");
        Object result = proxy.invokeSuper(obj, args);  // 调用父类(真实对象)方法
        System.out.println("后置增强");
        return result;
    }
}

// 2. 生成代理对象
public static void main(String[] args) {
    Enhancer enhancer = new Enhancer();
    enhancer.setSuperclass(UserServiceImpl.class);  // 设置被代理类
    enhancer.setCallback(new CglibProxy());        // 设置回调
    UserService proxy = (UserService) enhancer.create();  // 创建代理
    proxy.save();
}

原理:通过ASM库生成被代理类的子类,重写方法并插入增强逻辑。


通过理解静态代理与动态代理的实现和原理,可灵活选择适合场景的代理方式,高效实现功能增强和解耦。

相关推荐
秋月的私语1 小时前
如何快速将当前的c#工程发布成单文件
android·java·c#
天***88961 小时前
使用python写一个应用程序要求实现微软常用vc++功能排查与安装功能
java
代码充电宝2 小时前
LeetCode 算法题【简单】283. 移动零
java·算法·leetcode·职场和发展
caibixyy4 小时前
Spring Boot 整合 Redisson 实现分布式锁:实战指南
spring boot·分布式·后端
码事漫谈5 小时前
C++编程陷阱:悬空引用检测方法与防范指南
后端
码事漫谈5 小时前
缓存友好的数据结构设计:提升性能的关键技巧
后端
ccccczy_5 小时前
Spring Security 深度解读:JWT 无状态认证与权限控制实现细节
java·spring security·jwt·authentication·authorization·securityfilterchain·onceperrequestfilter
Lin_Aries_04215 小时前
容器化 Tomcat 应用程序
java·linux·运维·docker·容器·tomcat
sheji34165 小时前
【开题答辩全过程】以 springboot高校社团管理系统的设计与实现为例,包含答辩的问题和答案
java·spring boot·后端
zzywxc7876 小时前
大模型落地实践指南:从技术路径到企业级解决方案
java·人工智能·python·microsoft·golang·prompt