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 天前
别再被误导!try...catch性能大揭秘
后端
无巧不成书02181 天前
30分钟入门Java:从历史到Hello World的小白指南
java·开发语言
苍何1 天前
30分钟用 Agent 搓出一家跨境网店,疯了
后端
ssshooter1 天前
Tauri 2 iOS 开发避坑指南:文件保存、Dialog 和 Documents 目录的那些坑
前端·后端·ios
追逐时光者1 天前
一个基于 .NET Core + Vue3 构建的开源全栈平台 Admin 系统
后端·.net
程序员飞哥1 天前
90后大龄程序员失业4个月终于上岸了
后端·面试·程序员
zs宝来了1 天前
Playwright 自动发布 CSDN 的完整实践
java
吴声子夜歌1 天前
TypeScript——基础类型(三)
java·linux·typescript
GetcharZp1 天前
Git 命令行太痛苦?这款 75k Star 的神级工具,让你告别“合并冲突”恐惧症!
后端