1.为什么要使用代理

  代理可以在运行时创建一个实现了一组给定接口的新类。这种功能只有在编译时无法确定需要实现哪个接口时才有必要使用。

  假设有一个表示接口的Class对象(有可能只包含一个接口),它的确切类型在编译时无法知道,如果想要根据这个Class对象来构造一个实现这些接口的类,就需要使用newInstance方法或者反射找出类的构造器,但是,不能实例化一个接口,需要在程序处于运行状态时定义一个新类。

  代理机制可以解决这个问题,代理类可以在运行时创建新的类,这样的代理类能够实现执行的接口。并且代理类具有指定接口中的全部方法以及Object类中的全部方法。

  2.创建代理对象

  要想创建一个代理对象,需要使用Proxy类的newProxyInstance方法。

  这个方法有三个参数:

  • ClassLoader loader:类加载器,用null表示默认的类加载器。
  • Class<?>[] interfaces:Class对象数组,每个元素都是需要实现的接口
  • InvocationHandler h:调用处理器
public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h)

  调用处理器是实现了InvocationHandler接口的类的对象,这个接口中只有一个invoke方法,无论什么时候调用代理对象的方法,调用处理器的invoke方法都会被调用,并向其传递Method对象和原始的调用参数。

public Object invoke(Object proxy, Method m, Object[] args)

  以一个例子说明:

package proxy;

import java.lang.reflect.*;
import java.util.*; 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;
InvocationHandler handler = new TraceHandler(value);
Object proxy = Proxy.newProxyInstance(null, new Class[] { Comparable.class } , handler);
elements[i] = proxy;
} Integer key = new Random().nextInt(elements.length) + 1;
System.out.println("key: " + key); int result = Arrays.binarySearch(elements, key);
System.out.println("result: " + result); if (result >= 0) System.out.println(elements[result]);
}
} class TraceHandler implements InvocationHandler
{
private Object target; public TraceHandler(Object t)
{
target = t;
} public Object invoke(Object proxy, Method m, Object[] args) throws Throwable
{
// print implicit argument
System.out.print(target);
// print method name
System.out.print("." + m.getName() + "(");
// print explicit arguments
if (args != null)
{
for (int i = 0; i < args.length; i++)
{
System.out.print(args[i]);
if (i < args.length - 1) System.out.print(", ");
}
}
System.out.println(")"); // invoke actual method
return m.invoke(target, args);
}
}

  以一个例子来分析代码执行过程,首先for循环,创建了1000个Interger对象value,将这些value对象传递到处理器的构造函数中,然后构造一个代理类对象,将这些代理类对象放入到element对象数组中。然后构造一个随机的Integer对象,假设key值为104,然后Arrays.binarySearch(elements, key);在elements对象数组中寻找对象值为104的索引,这个时候就发生了一些很令人难人寻味的事情。

  这里就要看一下Arrays.binarySearch方法的具体实现:

    public static int binarySearch(Object[] a, Object key) {
return binarySearch0(a, 0, a.length, key);
}

 然后看一下binarySearch0方法的具体实现,注意参数传递关系,a是第一个参数,key是最后一个参数

    // Like public version, but without range checks.
private static int binarySearch0(Object[] a, int fromIndex, int toIndex,
Object key) {
int low = fromIndex;
int high = toIndex - 1; while (low <= high) {
int mid = (low + high) >>> 1;
@SuppressWarnings("rawtypes")
Comparable midVal = (Comparable)a[mid];
@SuppressWarnings("unchecked")
int cmp = midVal.compareTo(key);
if (cmp < 0)
low = mid + 1;
else if (cmp > 0)
high = mid - 1;
else
return mid; // key found
}
return -(low + 1); // key not found.
}

  注意这个方法中的下面的代码,同时注意参数对应关系

            Comparable midVal = (Comparable)a[mid];int cmp = midVal.compareTo(key);

  有点乱,将思路再次梳理一下:

  (1)Arrays.binarySearch(elements, key);其中,key是Integer对象104

  (2)调用binarySearch(elements,  104);返回binarySearch0(elements, 0, elements.length, 104);

  (3)然后开始调用binarySearch0(elements, 0, elements.length, 104);方法:mid=500,midVal是Object对象elements[500]采用强制类型转换得到的Comparable接口类型的对象,在int cmp = midVal.compareTo(104);这行代码执行的时候,由于数组中都是代理对象,而elements[500]也是(null, new Class[] { Comparable.class } , handler);且handler = new TraceHandler(value);中value是500对应的代理对象,当这个代理对象执行.compareTo(104)这个方法的时候,由于compareTo方法是Comparable接口中的唯一方法如果代理对象调用了这个方法,就会去执行hander调用处理器对象对应的invoke(Object proxy, Method m, Object[] args)方法,此时构造函数中的target就是value也就是500,elements[500].compareTo(104)对应到invoke方法中就是,proxy=elements[500],m=compareTo,args={104},然后就简单了,首先打印500,然后打印.compareTo(,然后打印args中所有的对象即104,然后打印),最后返回的是m.invoke(target, args);对应着elements[500].compareTo(104);也就是又回到了原来的方法中...最后二分法结束之后返回的result的值才是索引值并且大于0,然后进入System.out.println(elements[result]);然后又开始了...又开始了,

  System.out.println的实现是:

    public void println(Object x) {
String s = String.valueOf(x);
synchronized (this) {
print(s);
newLine();
}
}

  String.valueOf的实现是:

    public static String valueOf(Object obj) {
return (obj == null) ? "null" : obj.toString();
}

  这里就看出问题来了,toString方法也会被重定向到调用处理器上,这是为什么呢,明明toString都不属于Comparable接口,为什么还是要调用处理器。这是因为,即使不属于Comparable接口,toString方法也被代理了,因此就会...

  总的输出为:

key: 104
500.compareTo(104)
250.compareTo(104)
125.compareTo(104)
62.compareTo(104)
93.compareTo(104)
109.compareTo(104)
101.compareTo(104)
105.compareTo(104)
103.compareTo(104)
104.compareTo(104)
result: 103
104.toString()
104

  3.代理类的特性

  下面就要解释一下上面的toString方法到底是怎么回事。

  所有的代理类都扩展于Proxy类,一个代理类只有一个实例域--调用处理器,它定义在Proxy的父类中。为了履行代理对象的职责,所需要的任何附加数据都必须存储在调用处理器中。例如,代理Comparable对象时,TranceHandler包装了实际的对象。

  所有的代理类都覆盖了Object类中的方法toString、equals和hashCode。如同所有的代理方法一样,这些方法仅仅调用了调用处理器的invoke。Object类中的其他方法没有被重新定义。

  对于特定的类加载器和预设的一组接口来说,只能有一个代理类。即如果使用同一个类加载器和接口数组调用两次newProxyInstance方法的话,那么只能够得到同一个类的两个对象。

  代理类一定是public和final,如果代理类实现的所有接口都是public,代理类就不属于某个特定的包;否则,所有非公有的接口都必须属于同一个包,同时,代理类与属于这个包。

  可以通过Proxy类中的isProxyClass方法检测一个特定的Class对象是否代表着一个代理类。

  代理类真的是很麻烦呀。

Java基础(十四)代理(Proxy)的更多相关文章

  1. java基础(十四)-----详解匿名内部类——Java高级开发必须懂的

    在这篇博客中你可以了解到匿名内部类的使用.匿名内部类要注意的事项.匿名内部类使用的形参为何要为final. 使用匿名内部类内部类 匿名内部类由于没有名字,所以它的创建方式有点儿奇怪.创建格式如下: n ...

  2. Java基础(十四)--装箱、拆箱详解

    Java中基本数据类型都有相对应的包装类 什么是装箱?什么是拆箱? 在Java SE5之前,Integer是这样初始化的 Integer i = new Integer(10); 而在从Java SE ...

  3. Java实习生常规技术面试题每日十题Java基础(四)

    目录 1.String 和StringBuffer的区别. 2.数组有没有length()这个方法? String有没有length()这个方法? 3.final, finally, finalize ...

  4. 黑马程序员:Java基础总结----静态代理模式&动态代理

    黑马程序员:Java基础总结 静态代理模式&动态代理   ASP.Net+Android+IO开发 . .Net培训 .期待与您交流! 静态代理模式 public  class  Ts {   ...

  5. “全栈2019”Java第九十四章:局部内部类详解

    难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java第 ...

  6. “全栈2019”Java第十四章:二进制、八进制、十六进制

    难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java第 ...

  7. “全栈2019”Java第二十四章:流程控制语句中决策语句switch下篇

    难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java第 ...

  8. Bootstrap<基础十四> 按钮下拉菜单

    使用 Bootstrap class 向按钮添加下拉菜单.如需向按钮添加下拉菜单,只需要简单地在在一个 .btn-group 中放置按钮和下拉菜单即可.也可以使用 <span class=&qu ...

  9. Java基础十二--多态是成员的特点

    Java基础十二--多态是成员的特点 一.特点 1,成员变量. 编译和运行都参考等号的左边. 覆盖只发生在函数上,和变量没关系. Fu f = new Zi();System.out.println( ...

  10. Java基础十--接口

    Java基础十--接口 一.接口的定义和实例 /* abstract class AbsDemo { abstract void show1(); abstract void show2(); } 8 ...

随机推荐

  1. python实现感知机线性分类模型

    前言 感知器是分类的线性分类模型,其中输入为实例的特征向量,输出为实例的类别,取+1或-1的值作为正类或负类.感知器对应于输入空间中对输入特征进行分类的超平面,属于判别模型. 通过梯度下降使误分类的损 ...

  2. Mybaits-从零开始-Spring、Spring MVC、MyBatis整合(未万待续)

    Spring.Spring MVC.MyBatis整合(未万待续)

  3. eShopOnContainers学习系列(三):RabbitMQ消息总线实践

    今天研究了下eShopOnContainers里的RabbitMQ的使用,在项目里是以封装成消息总线的方式使用的,但是仍然是以其发布.订阅两个方法作为基础封装的,我们今天就来实际使用一下. 为了简单起 ...

  4. SpringBootSecurity学习(15)前后端分离版之 OAuth2.0简单示例

    OAuth2.0 OAuth 引入了一个授权层,用来分离两种不同的角色:客户端和资源所有者.客户端来申请资源,资源所有者同意以后,资源服务器可以向客户端颁发令牌.客户端通过令牌,去请求数据.也就是说, ...

  5. Shell之变量

    目录 Shell之变量 参考 变量命名规则 变量语法规范 四种变量类型 系统变量 特殊变量 变量的使用 变量内容的删除和替换 Shell之变量

  6. export import 的用法和注意之处

       1.整体引入: 会将若干export导出的内容组合成一个对象返回: import *as  api from  utils.https; api为自定义名称,可直接指定此文件中的某个方法,uti ...

  7. RocketMQ 源码学习笔记 Producer 是怎么将消息发送至 Broker 的?

    目录 RocketMQ 源码学习笔记 Producer 是怎么将消息发送至 Broker 的? 前言 项目结构 rocketmq-client 模块 DefaultMQProducerTest Roc ...

  8. Label的作用是什么,是怎么用的?

    label标签来定义表单控制间的关系,当用户选择该标签时,浏览器会自动将焦点转到标签相关的表单控件上. 如: <form> <label for="male"&g ...

  9. Flutter 修改SDK 路径出现的问题

    更换Flutter SDK 路径之后出现的问题. 解决方法:flutter packages upgrade. 替换: 替换这两个文件里的路径. 同时修改电脑配置文件里的路径: open -a Tex ...

  10. 如何在Java中创建数组列表

    为了在Java中存储动态大小的元素,我们使用了ArrayList.每当添加新元素时,它会自动增加它们的大小.ArrayList实现Java的List接口和Java的Collection的一部分. 由于 ...