java中动态代理

本文深入探讨了代理模式在软件设计中的应用,特别是在Android开发中的Binder跨进程通信。文章详细讲解了静态代理与动态代理的区别,以及如何在Java中实现这两种代理模式。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

代理模式的使用场景如下:

当无法或不想直接访问某个对象或访问对象存在困难时可以通过一个代理对象来间接访问,为了保证客户端使用的透明性,委托对象与代理对象需要实现相同的接口。

在 Android 代码中经常会看见代理模式的存在,尤其是在 Binder 跨进程通信方面。比如 Activity 与 ActivityManagerService 的通信,Activity 拿到的就是 ActivityManagerService 的代理对象 ActivityManagerProxy ,由这个代理对象去执行跨进程通信。

而 ActivityManagerProxy 和 ActivityManagerService 又实现了同样的接口 IActivityManager。ActivityManagerService 是抽象类 ActivityManagerNative 的子类,ActivityManagerNative 实现了 IActivityManager接口。

代理模式的作用就是为其他对象提供一种代理以控制对这个对象的访问。而在 Android 代码中,这个代理大多通过跨进程的方式来访问了。

静态代理

Java 的静态代理实现比较简单,就是一个代理对象持有了被代理对象的引用。

// 被代理的类
RealSubject real = new RealSubject();
// 静态代理
ProxySubject proxySubject = new ProxySubject(real);
proxySubject.visit();12345

另外,代理对象和被代理对象都要实现相同的接口或者继承相同的父类。

// 定义接口
public interface ISubject 
// 被代理对象实现接口
public class RealSubject implements  ISubject
// 代理对象实现接口
public class ProxySubject implements ISubject123456

由于是静态代理,我们的代码在运行前编译, Java 的 class文件就会创建,然而, 一旦需要代理的类多了,就需要为每一个类都编写一个代理类,也就是生成了多个class文件。这样创建的静态代理还不如直接编写代码了,毕竟都是要生成文件的。

因此,为了解决上述问题,就可以考虑在程序运行时动态地生成代理的对象,在编译阶段不需要知道谁代理了谁,而 Java 也提供了一个便捷的动态代理接口 InvocationHandler,实现该接口需要重写其调用方法invoke。

动态代理

Proxy 类提供了用于创建动态代理类和代理对象的静态方法,它也是所有动态代理类的父类。

如果在程序中为一个或多个接口动态地生成实现类,就可以使用 Proxy getProxyClass 方法来创建动态代理类。

如果需要为一个接口或多个接口动态地创建实例,也可以使用 Proxy newProxyInstance 方法来创建动态代理实例。

创建动态代理类

private static Class<?> getProxyClass(ClassLoader loader, Class<?>… interfaces)1

该方法创建一个动态代理类所对应的 Class 对象,该代理类将实现 interfaces所指定的多个接口,第一个参数ClassLoader指定生成动态代理类的类加载器。

定义如下接口,动态创建的代理类要实现该接口。

public interface SampleInterface {
void showMessage();
}123

通过 getProxyClass 方法创建动态代理类:

    Class proxyClass = Proxy.getProxyClass(
            SampleInterface.class.getClassLoader(), new Class[]{SampleInterface.class}
            );123

然后在通过反射去创建动态代理类的一个实例对象:

    // 得到的构造函数 getConstructor 带有参数,newInstance 也有参数
    SampleInterface proxy = (SampleInterface) proxyClass
            .getConstructor(new Class[]{InvocationHandler.class})
            .newInstance(new Object[]{handler});1234

此时实例化动态代理类的一个对象需要传递一个参数,它必须实现了 InvocationHandler 接口的参数。

InvocationHandler handler = new SampleInvocationHandler();

public class SampleInvocationHandler implements InvocationHandler {
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
System.out.printf(“Sample InvocationHandler \n”);
return null;
}
}123456789

我们通过 getProxyClass 创建了类,并且这个类实现了指定的接口,那么这些接口的调用就是由 InvocationHandler 来实现的。

在 invoke 方法里面,我们要根据 method 和 args 等参数来判断当前调用是对应接口的哪个方法,从而执行对应方法。

创建动态代理对象实例

public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h)1

该方法创建一个动态代理对象,该代理对象的实例实现了 interfaces指定的系列接口,执行代理对象的每个方法时都会被替换成InvocationHandler对象的invoke方法。

下面就是一个简单示例:

首先定义一个代理接口:

public interface IDynamicProxy {
void show();
}123

然后,通过 newProxyInstance 方法来创建代理对象实例,其中第一个参数是类加载器,第二个参数是要实现的接口,最后的 InvocationHandler 对象就是当调用代理对象实现的那个接口方法时,都会通过它的 invoke 方法来调用到具体的对应方法。

invoke 方法也是通过 method 和 args 参数来判断具体对应的接口方法是哪一个。

    IDynamicProxy proxy = (IDynamicProxy) Proxy.newProxyInstance(
            IDynamicProxy.class.getClassLoader(), new Class<?>[]{IDynamicProxy.class
    }, new InvocationHandler() {
        @Override
        public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
            System.out.printf("invoke\n");
            if (method.getReturnType() != Void.class){
                System.out.printf("return void");
            }
            return 1;
        }
    });
    proxy.show();12345678910111213

当通过 newProxyInstance 创建对象之后,就可以直接调用接口方法了。

小结

通过动态代理来创建代理类或者代理对象,有一个公共点就是都需要创建 InvocationHandler 类来执行具体的方法调用,并且在 invoke 方法里面通过 method 和 args 来区分具体调用的方法。


本文来自 zhying719 的CSDN 博客 ,全文地址请点击:https://blog.csdn.net/zhying719/article/details/82156094?utm_source=copy

<think>好的,用户想了解Java动态代理的应用场景和示例。首先,我需要回忆一下动态代理的基本概念,确保自己理解正确。动态代理主要是在运行时动态生成代理类,而不需要像静态代理那样手动编写代理类。这通常通过Javajava.lang.reflect.Proxy和InvocationHandler来实现。 接下来,用户需要的是应用场景和示例。根据已有的引用资料,比如引用[1]提到了AOP、远程方法调用、延迟加载和事务管理,引用[5]也提到了框架中的应用。需要把这些场景整理出来,每个场景配一个具体的例子,最好是代码示例。 然后要注意用户提到的格式要求:行内数学表达式用$...$,独立公式用$$...$$。不过这个问题可能不涉及数学公式,但代码块需要用```python或者其他语言标签。不过用户的问题是关于Java的,所以代码示例应该用Java。但原示例中的代码用了Python,可能需要调整。 用户还要求生成相关问题,所以需要最后添加相关问题部分。同时,引用标识要自然地添加在引用的段落末尾,比如引用[^1]、[^2]等。需要确保每个引用对应的内容正确,比如引用[1]关于AOP的应用,引用[5]关于框架中的使用。 另外,回答的结构要清晰,分步骤或分点说明。用户可能希望看到每个应用场景的具体解释和代码示例,这样更容易理解。比如,AOP的场景可以结合Spring框架中的事务管理,但可能需要简化,用更基础的例子来说明动态代理如何实现方法增强。 还要注意使用中文回答,保持语言流畅自然。需要检查是否有遗漏的应用场景,比如远程调用(如RMI)、延迟加载(如Hibernate的延迟加载)和事务管理,这些都需要例子支持。 最后,生成的相关问题应该与动态代理相关,比如原理、与静态代理的区别、性能比较等,确保问题有针对性,帮助用户进一步学习。</think>### Java动态代理的应用场景及示例 Java动态代理主要用于**运行时动态生成代理对象**,无需手动编写代理类,适用于以下典型场景: --- #### 1. **面向切面编程(AOP)** 动态代理可在方法执行前后插入通用逻辑(如日志、事务、权限校验)。例如,为所有Service层方法添加事务控制: ```java // 接口 public interface UserService { void saveUser(User user); } // 动态代理实现事务管理 public class TransactionHandler implements InvocationHandler { private Object target; public TransactionHandler(Object target) { this.target = target; } @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { try { System.out.println("开启事务"); Object result = method.invoke(target, args); System.out.println("提交事务"); return result; } catch (Exception e) { System.out.println("回滚事务"); throw e; } } } // 使用代理 UserService userService = (UserService) Proxy.newProxyInstance( UserService.class.getClassLoader(), new Class[]{UserService.class}, new TransactionHandler(new UserServiceImpl()) ); userService.saveUser(user); // 代理对象自动管理事务 ``` 此模式在Spring框架的事务管理中广泛应用[^5]。 --- #### 2. **远程方法调用(RPC/RMI)** 动态代理可隐藏网络通信细节,例如Dubbo等框架通过代理封装远程调用: ```java // 客户端调用远程服务接口 public interface OrderService { Order getOrderById(int id); } // 代理类处理网络请求 public class RpcProxy implements InvocationHandler { @Override public Object invoke(Object proxy, Method method, Object[] args) { // 序列化参数并发送网络请求 return sendRequest(method.getName(), args); } } // 客户端透明调用 OrderService orderService = (OrderService) Proxy.newProxyInstance(...); Order order = orderService.getOrderById(123); // 实际触发远程调用 ``` --- #### 3. **延迟加载(Lazy Loading)** 动态代理可延迟资源加载,例如Hibernate中关联对象的延迟加载: ```java public class LazyLoader implements InvocationHandler { private Object realObject; private Supplier<Object> loader; public LazyLoader(Supplier<Object> loader) { this.loader = loader; } @Override public Object invoke(Object proxy, Method method, Object[] args) { if (realObject == null) { realObject = loader.get(); // 首次调用时加载真实对象 } return method.invoke(realObject, args); } } // 使用代理延迟加载大对象 ImageService imageService = (ImageService) Proxy.newProxyInstance(...); // 真实对象仅在调用时加载 imageService.getHighResolutionImage(); ``` --- #### 4. **接口适配与装饰器模式** 动态代理可统一处理多个接口的公共逻辑,例如为多个Service添加性能监控: ```java public class MetricsHandler implements InvocationHandler { private Object target; @Override public Object invoke(Object proxy, Method method, Object[] args) { long start = System.currentTimeMillis(); Object result = method.invoke(target, args); System.out.println("方法耗时: " + (System.currentTimeMillis() - start) + "ms"); return result; } } // 为任意Service接口添加监控 UserService userService = Proxy.newProxyInstance(...); ProductService productService = Proxy.newProxyInstance(...); ``` --- ###
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值