实现动态代理

     

动态代理(Dynamic Proxy):相比前一篇文章所实现的静态代理,动态代理具有更强的灵活性,因为它不用在我们设计实现的时候就指定某一个代理类来代理哪一个被代理对象,我们可以把这种指定延迟到程序运行时由JVM来实现。

所谓“设计实现的时候的代码”是指不是调用者的代码。而是提供给调用者的代码。所谓动态,在调用的时候,才去指定谁和谁联系。

 
我们知道,所谓代理,就是需要代理类和被代理类有相同的对外接口或者说成服务,所以代理类一般都必须实现了所有被代理类已实现的接口,因为接口就是制定了一系列对外服务的标准。
 
正因为动态代理有这样灵活的特性,所以我们在设计动态代理类(DynamicProxy)时不用显式地让它实现与真实主题类(RealSubject)相同的接口(interface),而是把这种实现推迟到运行时。
 
为了能让DynamicProxy类能够在运行时才去实现RealSubject类已实现的一系列接口并执行接口中相关的方法操作,需要让DynamicProxy类实现JDK自带的java.lang.reflect.InvocationHandler接口,该接口中的invoke()方法能够让DynamicProxy实例在运行时调用被代理类的“对外服务”,即调用被代理类需要对外实现的所有接口中的方法,也就是完成对真实方法的调用,Java帮助文档中称这些真实方法为处理程序。
 
按照上面所述,我们肯定必须先把被代理类RealSubject已实现的所有interface都加载到JVM中,不然JVM怎么能够找到这些方法呢?明白了这个道理,那么我们就可以创建一个被代理类的实例,获得该实例的类加载器ClassLoader
 
所谓的类加载器ClassLoader,就是具有某个类的类定义,即类的内部相关结构(包括继承树、方法区等等)。
 
更重要的是,动态代理模式可以使得我们在不改变原来已有的代码结构的情况下,对原来的“真实方法”进行扩展、增强其功能,并且可以达到控制被代理对象的行为的目的。请详看下面代码中的DynamicProxy类,其中必须实现的invoke()方法在调用被代理类的真实方法的前后都可进行一定的特殊操作。这是动态代理最明显的优点。
代码:

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;


//抽象主题类,这里不能用abstract抽象类,一定要是interface
interface AbstractSubject {
    public abstract void request();
}


// 真实主题类,即被代理类
class RealSubject implements AbstractSubject {
    public void request() {
        System.out.println("RealSubject's request() ...");
    }
}


// 动态代理类,实现InvocationHandler接口
class DynamicProxy implements InvocationHandler {
    // 被代理类的实例
    Object obj = null;
    // 将被代理者的实例传进动态代理类的构造函数中
    public DynamicProxy(Object obj) {
        this.obj = obj;
    }
    /**       * 覆盖InvocationHandler接口中的invoke()方法
     *        * 更重要的是,动态代理模式可以使得我们在不改变原来已有的代码结构
     * 的情况下,对原来的“真实方法”进行扩展、增强其功能,并且可以达到
     * 控制被代理对象的行为,下面的before、after就是我们可以进行特殊
     * 代码切入的扩展点了。       */
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        /*           * before :doSomething();           */   //扩展
        Object result = method.invoke(this.obj, args);
        /*           * after : doSomething();           */    //扩展
        return result;      }
}


// 测试类
public class Client {
    public static void main(String[] args) {
        //满眼的反射机制
        
        // 被代理类的实例
        AbstractSubject realSubject = new RealSubject();
        // 获得被代理类的类加载器,使得JVM能够加载并找到被代理类的内部结构,以及已实现的interface
        ClassLoader loader = realSubject.getClass().getClassLoader();
        // 获得被代理类已实现的所有接口interface,使得动态代理类的实例
        Class<?>[] interfaces = realSubject.getClass().getInterfaces();

        //max-------------------------------------
//        for(int i = 0;i < interfaces.length;i++)
//        {
//            Class inter = interfaces[i];
//            String info = inter.getName();
//            System.out.println(info);
//        }
        //max-------------------------------------

        // 用被代理类的实例创建动态代理类的实例,用于真正调用处理程序
        InvocationHandler handler = new DynamicProxy(realSubject);   //这时指定生成它的代理
        /*           * loader : 被代理类的类加载器
         * interfaces :被代理类已实现的所有接口,而这些是动态代理类要实现的接口列表
         * handler : 用被代理类的实例创建动态代理类的实例,用于真正调用处理程序
         *            * return :返回实现了被代理类所实现的所有接口的Object对象,即动态代理,需要强制转型
         */
        //获得代理的实例
        AbstractSubject proxy = (AbstractSubject) Proxy.newProxyInstance(
                loader, interfaces, handler); //通过InvocationHandler和Proxy创建代理类,有点类似于工厂
        proxy.request();
        //打印出该代理实例的名称
        System.out.println(proxy.getClass().getName());
    }
}
 
测试结果:
RealSubject's request() ...
DesignPattern.proxy.dynamicProxy.$Proxy0
话又说回来,动态代理机制确实很灵活,或者说很智能,但是这是运用到了Java中的反射机制,而反射机制又与JVM中栈区、堆区、方法区等底层细节以及类的加载、生命周期等知识相关,要完全理解相当不容易,看来还有很长的路要走呢!
 
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Java中可以通过使用反射机制和动态代理实现动态代理动态代理是一种在运行时创建代理对象的机制,它可以在不修改原始类的情况下,对方法进行增强或拦截。 Java中实现动态代理的步骤如下: 1. 定义一个接口:首先需要定义一个接口,该接口包含需要被代理的方法。 2. 创建一个实现InvocationHandler接口的类:该类实现了InvocationHandler接口,并重写了invoke方法。在invoke方法中,可以对原始方法进行增强或拦截。 3. 使用Proxy类创建代理对象:通过调用Proxy类的静态方法newProxyInstance,传入ClassLoader、接口和InvocationHandler对象,可以创建一个代理对象。 4. 调用代理对象的方法:通过调用代理对象的方法,实际上会调用InvocationHandler中的invoke方法。 下面是一个示例代码: ```java import java.lang.reflect.InvocationHandler; import java.lang.reflect.Method; import java.lang.reflect.Proxy; // 定义接口 interface Hello { void sayHello(); } // 实现InvocationHandler接口 class MyInvocationHandler implements InvocationHandler { private Object target; public MyInvocationHandler(Object target) { this.target = target; } @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { // 在调用原始方法前进行增强 System.out.println("Before invoking sayHello()"); // 调用原始方法 Object result = method.invoke(target, args); // 在调用原始方法后进行增强 System.out.println("After invoking sayHello()"); return result; } } public class DynamicProxyExample { public static void main(String[] args) { // 创建原始对象 Hello hello = new HelloImpl(); // 创建InvocationHandler对象 InvocationHandler handler = new MyInvocationHandler(hello); // 创建代理对象 Hello proxy = (Hello) Proxy.newProxyInstance( hello.getClass().getClassLoader(), hello.getClass().getInterfaces(), handler); // 调用代理对象的方法 proxy.sayHello(); } } // 原始类 class HelloImpl implements Hello { @Override public void sayHello() { System.out.println("Hello, World!"); } } ``` 以上代码中,通过创建一个实现InvocationHandler接口的类MyInvocationHandler,并在invoke方法中对原始方法进行增强。然后使用Proxy类的newProxyInstance方法创建代理对象,并传入ClassLoader、接口和InvocationHandler对象。最后通过调用代理对象的方法,实际上会调用MyInvocationHandler中的invoke方法,从而实现动态代理
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值