【Java学习笔记(七十)】之代理的介绍

本文介绍了Java代理机制的概念,强调了代理类在运行时创建以实现未知接口的能力。调用处理器作为中介,在调用代理对象方法时执行特定操作。通过`Proxy.newProxyInstance`创建代理对象,并给出了一个用于跟踪方法调用的`TraceHandler`实例。代理在远程调用、UI绑定和调试等方面有广泛应用。最后,文章展示了如何使用代理进行二分查找并追踪`compareTo`方法的调用。
摘要由CSDN通过智能技术生成

本文章由公号【开发小鸽】发布!欢迎关注!!!


老规矩–妹妹镇楼:

一. 代理的意义

       当我们想要构造一个类的对象时,这个类实现了一个或者多个接口,在编译时可能是不知道这些接口到底是什么。普通的类可以使用newInstance方法或者使用反射找出构造器,但是接口是无法直接实例化的,需要在运行的程序中定义一个新的实现类。为了解决这个问题,代理机制出现了,代理类可以在运行时创建一个全新的类,这样的代理类能够实现你指定的接口。


二. 调用处理器

       调用处理器的作用是在调用代理对象的方法时,都会转向调用处理器,调用其中的invoke方法。调用处理器是一个实现了InvocationHandler接口的类对象,这个接口只有一个invoke方法:

Object invoke(Object proxy, Method method, Object[] args)

三. 创建代理对象

       要想创建一个代理对象,需要使用Proxy类的newProxyInstance方法,这个方法有三个参数:

类加载器
Class对象数组
调用处理器
Object proxy = Proxy.newProxyInstance(ClassLoader.getSystemClassLoader(), new Class[]{Comparable.class}, handler);

四. 使用代理的场景

  1. 将方法调用路由到远程服务器

  2. 在运行的程序中将用户界面时间和动作关联起来

  3. 为了调试,跟踪方法调用


五. 调用代理对象的方法

       使用代理对象跟踪一个二分查找,首先在elements数组中添加代理,然后调用Arrays类的binarySearch方法在数组中查找一个随机整数,在binarySearch方法中就会有以下的调用:

if(elements[i].compareTo(key)<0)

       由于elements数组中填充了代理对象,所以compareTo调用了调用处理器中的invoke方法,可以看出,只要调用了代理对象的方法,都会调用invoke方法。


六. 代理类的特性

       代理类是在程序运行过程中动态创建的,一旦被创建,它们就变成了常规类,与虚拟机中的其他类没有什么区别。所有的代理类都会扩展Proxy类,且一个代理类中只有一个实例字段,就是调用处理器,完成代理对象任务所需要的所有额外数据都需要存储在调用处理器之中。

       所有的代理类都会覆盖Object类的toString, equals和hashCode方法,这些方法都会调用处理器的invoke方法。代理类是没有定义名字的,虚拟机将会给它自动生成一个以字符串 $Proxy开头的类名。

       使用同一个类加载器和预设的接口,只能得到一个代理类。如果使用同一个类加载器和接口数组调用两次newProxyInstance方法, 将得到同一个类的两个对象。使用Proxy类的isProxyClass方法可以检测出一个特定的Class对象是否表示一个代理类。


七. 代理实例

1. 调用处理器

package proxy;

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

public class TraceHandler implements InvocationHandler {

    private Object target;
    public TraceHandler(Object t){
        target = t;
    }

    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println(target);
        System.out.println("." + method.getName() + "(");
        if(args != null){
            for(int i = 0; i < args.length; ++i){
                System.out.println(args[i]);
                if(i < args.length-1){
                    System.out.println(", ");
                }
            }
        }
        System.out.println(")");
        return method.invoke(target, args);
    }
}

2. 创建代理对象并且调用

package proxy;

import java.lang.reflect.Proxy;
import java.util.Arrays;
import java.util.Random;

public class ProxyTest {
    public static void main(String[] args) {
        Object[] elements = new Object[1000];
        for(int i = 0; i < elements.length; ++i){
            Integer value = i+1;
            TraceHandler handler = new TraceHandler(value);
            Object proxy = Proxy.newProxyInstance(ClassLoader.getSystemClassLoader(), new Class[]{Comparable.class}, handler);
            elements[i] = proxy;
        }
        Integer key = new Random().nextInt(elements.length)+1;
        int result = Arrays.binarySearch(elements, key);
        if(result >= 0){
            System.out.println(elements[result]);
        }
    }
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值