百度360必应搜狗淘宝本站头条
当前位置:网站首页 > IT精选 > 正文

Java代理模式详解:智能中介的编程艺术

ccwork 2025-04-10 21:15 11 浏览

一、生活场景中的代理思维

想象您要租房子,但不想直接与房东打交道,这时房产中介就发挥作用了:

  1. 1. 中介帮您筛选房源(访问控制)
  2. 2. 签约前验证房东资质(预处理)
  3. 3. 协助办理合同手续(功能增强)
  4. 4. 处理后续维修问题(后置处理)

这种中介服务思维正是代理模式的核心——在不修改原始对象的前提下,通过代理对象控制访问并增强功能。在编程领域,代理模式常用于远程调用、权限验证、日志记录等场景。


二、代理模式核心架构

1. 三大核心角色

角色

作用

现实类比

Subject

定义代理与目标对象的统一接口

租房标准流程

RealSubject

实际业务执行者(被代理对象)

真实房东

Proxy

中介层,控制并增强目标对象访问

房产中介

2. 工作流程示意

客户端 → 代理对象 → 预处理 → 目标对象 → 后处理 → 返回结果

三、静态代理:手工打造的专属代理

1. 代码实现示例

// 文件存储接口
interface FileStorage {
    voidupload(String filename);
}

// 真实对象
class CloudStorage implements FileStorage {
    publicvoidupload(String filename) {
        System.out.println("上传文件至云端: " + filename);
    }
}

// 代理类
class StorageProxy implements FileStorage {
    private CloudStorage target;
    
    public StorageProxy() {
        this.target = newCloudStorage();
    }

    @Override
    public void upload(String filename) {
        if(checkAccess()) {                  // 权限验证
            encryptFile(filename);           // 文件加密
            target.upload(filename);         // 调用真实方法
            logOperation(filename);          // 记录日志
        }
    }
    
    private boolean checkAccess() {
        // 模拟权限验证逻辑
        returntrue;
    }
}

2. 静态代理特点

o 优点:结构清晰,适合简单场景
o 缺点:每个接口需单独编写代理类,维护成本高
o 适用场景:代理逻辑固定且目标类较少的情况


四、动态代理:运行时生成的智能中介

1. JDK动态代理(基于接口)

public class AuditHandler implements InvocationHandler {
    private Object target;
    
    public AuditHandler(Object target) {
        this.target = target;
    }

    @Override
    public Object invoke(Object proxy, Method method, Object[] args)throws Throwable {
        System.out.println("【审计】操作开始: " + method.getName());
        Objectresult= method.invoke(target, args);
        System.out.println("【审计】操作耗时: " + (System.currentTimeMillis() - start) + "ms");
        return result;
    }
}

// 使用示例
FileStorage realService= new CloudStorage();
FileStorage proxy= (FileStorage) Proxy.newProxyInstance(
    realService.getClass().getClassLoader(),
    realService.getClass().getInterfaces(),
    new AuditHandler(realService)
);
proxy.upload("data.zip");

2. CGLIB代理(基于继承)

Enhancer enhancer=new Enhancer();
enhancer.setSuperclass(CloudStorage.class);
enhancer.setCallback((MethodInterceptor)(obj, method, args, proxy) -> {
    System.out.println("CGLIB预处理");
    Object result= proxy.invokeSuper(obj, args);
    System.out.println("CGLIB后处理");
    return result;
});
CloudStorage proxy= (CloudStorage) enhancer.create();

3. 动态代理对比

维度

JDK动态代理

CGLIB代理

技术基础

接口代理

继承代理

性能

调用较快

生成较慢但执行快

限制

必须实现接口

无法代理final类/方法

应用场景

Spring AOP默认方式

MyBatis Mapper代理


五、典型应用场景

1. 远程方法调用(RPC)

通过代理对象隐藏网络通信细节,如Dubbo框架的远程服务调用。

2. 事务管理

Spring使用AOP代理实现声明式事务:

@Transactional
public void transferMoney() {
    // 数据库操作
}

3. 延迟加载

MyBatis的懒加载机制通过代理实现:

// 当首次调用getOrders()时才执行SQL查询
User user = userMapper.findById(1);
List orders = user.getOrders(); 

4. 安全控制

权限验证代理示例:

public Object invoke(...) {
    if(!SecurityUtils.hasPermission(method.getName())) {
        throw new AccessDeniedException();
    }
    return method.invoke(target, args);
}

六、模式对比与选型

设计模式

核心区别

典型场景

代理模式

控制访问,功能增强

权限/日志/远程调用

装饰器模式

递归增强对象功能

IO流处理

适配器模式

接口转换

旧系统兼容

外观模式

简化复杂系统调用

微服务网关

选型建议
o 需要接口代理 → JDK动态代理
o 需要类代理 → CGLIB
o 简单固定逻辑 → 静态代理
o 复杂通用功能 → 动态代理


七、最佳实践指南

1. 性能优化技巧

o 缓存代理对象避免重复创建
o 优先使用接口代理(JDK动态代理)
o 对高频调用方法做字节码增强

2. 常见问题解答

Q:代理模式会导致性能下降吗?
现代JVM对反射调用有优化,在Spring等框架中代理性能损耗可控制在5%以内。

Q:如何选择动态代理实现?
o Spring Boot 2.x默认使用CGLIB
o 需要代理final类时可考虑ByteBuddy

Q:代理对象能被二次代理吗?
可以,但要注意调用链深度,过度代理会影响可维护性。


八、从原理到实践

1. 动态代理底层原理

JDK动态代理通过以下步骤生成$Proxy0.class

  1. 1. 根据接口生成字节码
  2. 2. 使用类加载器载入JVM
  3. 3. 创建代理对象实例

查看生成的代理类:

System.setProperty("sun.misc.ProxyGenerator.saveGeneratedFiles", "true");

2. Spring AOP实现揭秘

Spring通过以下组合实现AOP:

  1. 1. JDK动态代理(接口场景)
  2. 2. CGLIB(类代理场景)
  3. 3. AspectJ注解声明切面逻辑

掌握代理模式的关键在于理解"控制访问"的核心思想,这种编程范式已渗透到Java生态的各个角落。从Spring框架的事务管理到MyBatis的Mapper代理,从RPC调用到微服务网关,代理模式如同无形的纽带,将系统各模块优雅地连接在一起。通过合理运用代理模式,开发者可以构建出更灵活、更安全、更易维护的应用程序架构。

相关推荐

二十三、Java类与对象简介(java第十一章类和对象)

在Java编程语言中,类(Class)和对象(Object)是面向对象编程(OOP)的核心概念。描述类类是Java程序的基本组成单元,是对象的模板。类定义了对象的属性和方法。属性是对象的状态信息,而方...

设计模式-结构型-代理模式(proxy)

1.概念需要给对象提供一个代理以控制对该对象的访问,访问对象不适合或者不能直接引用目标对象,代理对象作为访问对象和目标对象之间的中介;根据代理类生成时机不同,分为静态代理和动态代理;静态代理代理类在编...

深度解析设计模式七大原则之——里氏替换原则

临近端午节,各位读者,你们假期行程安排好了吗?“菜鸟”已经做好决定了,谁都不能阻拦(产品经理也不行),“菜鸟”要好好在家休息三天。最近实在是太累了,一直在疯狂的加班。好了好了言归正传,开始我们的正文。...

Java代理模式详解:智能中介的编程艺术

一、生活场景中的代理思维想象您要租房子,但不想直接与房东打交道,这时房产中介就发挥作用了:1.中介帮您筛选房源(访问控制)2.签约前验证房东资质(预处理)3.协助办理合同手续(功能增强)4.处...

哪个创意最能打动你? 为你欣赏的“创意之星”投一票

这一期的《超级课堂·暑期特别活动》将评出5位“创意之星”,获得价值2000元的奖品。本期我们选登了部分中小学生在昙华林留下的创意作品,欢迎为最能打动你的作品投上一票。大众评审目前采取微信投票:扫描二维...

Netty基础—6.Netty实现RPC服务(netty reactor)

大纲1.RPC的相关概念2.RPC服务调用端动态代理实现3.Netty客户端之RPC远程调用过程分析4.RPC网络通信中的编码解码器5.Netty服务端之RPC服务提供端的处理6.RPC服务调用端实现...

静态代理和动态代理(静态代理和动态代理的优缺点)

1.什么是代理很多人肯定听过和看到过飞机票代理点,火车票代理点。那这些代理点干得事情就是帮航空公司,火车站出售火车票的工作。它们算是一个中间商。实际的服务不是由它们提供。而是由真正的服务商提供。通过这...

Java反射机制与Spring动态代理深度解析

一、Java反射机制原理剖析1.1反射的本质与实现基础Java反射(Reflection)是Java语言的核心特性,允许程序在运行时:动态加载类获取类结构元数据操作类属性和方法关键技术支撑:java...

Java 代理模式详解(java代理原理)

1.代理模式代理模式是一种比较好理解的设计模式。简单来说就是我们使用代理对象来代替对真实对象(realobject)的访问,这样就可以在不修改原目标对象的前提下,提供额外的功能操作,扩展目标对象...

SpringBoot全局异常处理:如何优雅应对多系统多格式错误响应需求

SpringBoot全局异常处理:如何优雅应对多系统多格式错误响应需求引言部分在微服务架构中,你是否曾为处理不同外部系统的异常响应而头痛?A系统要求返回JSON格式的:{code:1001,mes...

3分钟吃透代理技术!(代理一般都是怎么做)

最近有学员问了我一些问题,什么是代理,又该在什么地方使用。结合之前的讨论,这篇文章我们一起细致的讲解一下关于代理的一些问题。在Java中,代理通常分为两类:静态代理动态代理两者技术实现是不一样的,...

苏州网络维护 | 学习网络维护,从哪入手

我们想要学习网络维护,从哪入手呢?先带大家了解下网络维护1.培养基础知识:建立对计算机网络基本原理的理解。学习计算机网络的基础概念,如IP地址、子网掩码、路由器、交换机、协议等。2.学习网络技术:深入...

CAD如何快速一键编号?(cad如何一次性全部编号)

cad一键自动编号。·第一步,在命令行数abh空格。·第二步,打开自动编号对话框,选择用默认的图言编号,编号文字的高度根据图纸的大小设置零点八。当然如果图纸很大,设置比如十一百的有可能数字编号,这点很...

职场新人必知的10个高效工作法,助你快速升职加薪

初入职场,面对繁杂的工作任务和陌生的职场环境,如何才能快速适应并脱颖而出?以下是10个高效工作法,帮助职场新人提升工作效率,快速实现升职加薪的目标。---1.制定每日工作计划-推荐理由:每天开始...

《魔导英雄传说》新手攻略,快速升级,礼包码,最强阵容排列

《魔导英雄传说》新手攻略一、武将培养指南英雄选择与培养:英雄品质分为SSR、SR、A,优先培养SSR英雄,如张飞、孙尚香等核心英雄阵营搭配:同阵营上阵英雄越多,战力加成越高,建议优先培养同一阵营的英雄...