动态代理可以提供对另一个对象的访问,同时隐藏实际对象的具体事实,代理对象对客户隐藏了实际对象。动态代理可以对请求进行其他的一些处理,在不允许直接访问某些类,或需要对访问做一些特殊处理等,这时候可以考虑使用代理。

目前 Java 开发包中提供了对动态代理的支持,但现在只支持对接口的实现。

主要是通过 java.lang.reflect.Proxy 类和 java.lang.reflect.InvocationHandler 接口。
   Proxy 类主要用来获取动态代理对象,InvocationHandler 接口用来约束调用者行为。

“写一个 ArrayList 类的代理,其内部实现和 ArrayList 中完全相同的功能,并可以计算每个方法运行的时间。”这是一份考题上的题目,没有答案,来看下实现:

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.TimeUnit;
/**
 * -----------------------------------------
 * @描述  TODO
 * @作者  吴愿涛
 * @邮箱  manwuyuantao@qq.com
 * @日期  2012-11-9
 * -----------------------------------------
 */
public class ProxyApp {
    public static void main(String[] args){
       
        //ArrayList代理,通过代理计算每个方法调用所需时间
        List< Integer> arrayListProxy = (List< Integer>)Proxy.newProxyInstance(
      /*定义代理类的类加载器,用于创建代理对象,不一定必须是ArrayList,也可以是其他的类加载器*/
            ArrayList.class.getClassLoader(),  
            ArrayList.class.getInterfaces(),     /*代理类要实现的接口列表*/
            new InvocationHandler() {            /*指派方法调用的调用处理程序,这里用了匿名内部类*/
               
                private ArrayList< Integer> target = new ArrayList< Integer>(); //目标对象(真正操作的对象)
                /**
                 * 在代理实例上处理方法调用并返回结果
                 * @param proxy     代理对象(注意不是目标对象)
                 * @param method  被代理的方法
                 * @param args         被代理的方法的参数集
                 * @return  返回方法调用结果
                 */
                @Override
                public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
                   
                    long beginTime = System.currentTimeMillis();  //开始时间
                    TimeUnit.MICROSECONDS.sleep(1);
                    Object obj = method.invoke(target, args);          //实际调用的方法,并接受方法的返回值
                    long endTime = System.currentTimeMillis();   //结束时间
                    System.out.println("[" + method.getName() + "] spend " + (endTime - beginTime) + " ms");
                    return obj;   //返回实际调用的方法的返回值
                   
                }
               
            }
        );
        arrayListProxy.add(2);
        arrayListProxy.add(4);
        System.out.println("--------- 迭代 ---------");
        for(int i : arrayListProxy){
            System.out.print(i + "\t");
        }
    }
}
后台打印输出结果:
[add] spend 2 ms
[add] spend 1 ms
--------- 迭代 ---------
[iterator] spend 1 ms
2 4
从代码上来看,用到了匿名内部类,这样一来,InvocationHandler 只能用一次,
如果多个地方都需要用到这样一个相同的 InvocationHandler,可以将其抽象出来成为一个单独的类:
package test;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.util.concurrent.TimeUnit;
public 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 {
       
        long beginTime = System.currentTimeMillis();
        TimeUnit.MICROSECONDS.sleep(1);
        Object obj = method.invoke(target, args);
        long endTime = System.currentTimeMillis();
        System.out.println("[" + method.getName() + "] spend " + (endTime - beginTime) + " ms");
        return obj;
       
    }
}

客户端调用改成:

package example;
import java.lang.reflect.Proxy;
import java.util.ArrayList;
import java.util.List;
public class ProxyApp {
    public static void main(String[] args){
       
        //ArrayList代理,通过代理计算每个方法调用所需时间
        List< Integer> arrayListProxy = (List< Integer>)Proxy.newProxyInstance(
  /*定义代理类的类加载器,用于创建代理对象,不一定必须是ArrayList,也可以是其他的类加载器*/
            ArrayList.class.getClassLoader(),    
            ArrayList.class.getInterfaces(),       /*代理类要实现的接口列表*/
   /*指派方法调用的调用处理程序,这里用了匿名内部类*/
            new MyInvocationHandler(new ArrayList< Integer>())  
        );
        arrayListProxy.add(2);
        arrayListProxy.add(4);
        System.out.println("--------- 迭代 ---------");
        for(int i : arrayListProxy){
            System.out.print(i + "\t");
        }
    }
}

从上面代码看来,客户端知道代理的实际目标对象,还知道怎么样去创建这样一个代理对象,
如果想把这些信息全部对客户端隐藏起来,可以将这些代码挪到一个类中,将它们封装起来:
package example;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.TimeUnit;
public class ProxyUtil {
    public enum ArrayListProxy {
        PROXY;
       
        private Object target;
       
        ArrayListProxy(){
            this.target = new ArrayList< Object>();
        }
       
  public List getInstance(){
           
    return (List)Proxy.newProxyInstance(ArrayList.class.getClassLoader(), ArrayList.class.getInterfaces(),
                    new InvocationHandler() {
                       
               @Override
      public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
                           
        long beginTime = System.currentTimeMillis();
        TimeUnit.MICROSECONDS.sleep(1);
        Object obj = method.invoke(target, args);
        long endTime = System.currentTimeMillis();
        System.out.println("[" + method.getName() + "] spend " + (endTime - beginTime) + " ms");
           return obj;
                           
          }
        });
      }
    }
}

客户端调用改成:

package example;
import java.util.List;
import example.ProxyUtil.ArrayListProxy;
/**
 * -----------------------------------------
 * @描述  TODO
 * @作者  吴愿涛
 * @邮箱  manwuyuantao@qq.com
 * @日期  2012-11-9
 * -----------------------------------------
 */
public class ProxyApp {
    public static void main(String[] args){
       
        List< Integer> arrayListProxy = ArrayListProxy.PROXY.getInstance();
        arrayListProxy.add(2);
        arrayListProxy.add(4);
        System.out.println("--------- 迭代 ---------");
        for(int i : arrayListProxy){
            System.out.print(i + "\t");
        }
       
    }
}
上面代码中用到了枚举 enum,如果不想用枚举,就改用普通类来实现就行了。

转载于:https://www.cnblogs.com/manwuyuantao/archive/2012/11/09/2763106.html

写一个ArrayList类的动态代理类相关推荐

  1. 设计一个事务增强的动态代理类, 对持久层的用户的CRUD操作进行事务增强 即:

    动态代理类: package com.bj169.invoke;import com.bj169.dao.UserDao; import com.bj169.util.HibernateUtil; i ...

  2. 【转】浅析动态代理类实现过程

    代理模式 代理模式的作用是:为其他对象提供一种代理以控制对这个对象的访问.在某些情况下,一个客户不想或者不能直接引用另一个对象,而代理对象可以在客户端和目标对象之间起到中介的作用. 代理模式一般涉及到 ...

  3. 笔试题:写一个ArrayList 的动态代理类

    package com.zhangxueliang.demo;import java.lang.reflect.InvocationHandler; import java.lang.reflect. ...

  4. spring AbstractBeanDefinition创建bean类型是动态代理类的方式

    1.接口 Class<?> resourceClass 2.获取builder BeanDefinitionBuilder builder = BeanDefinitionBuilder. ...

  5. 初看Mybatis 源码 (二) Java动态代理类

    先抛出一个问题,用过Mybatis的都知道,我们只需要定义一个Dao的接口,在里面写上一些CRUD相关操作,然后配置一下sql映射文件,就可以达到调用接口中的方法,然后执行sql语句的效果,为什么呢? ...

  6. .Net基础——程序集与CIL HttpClient封装方法 .Net Core 编码规范 C#中invoke和beginInvoke的使用 WebServeice 动态代理类...

    .Net基础--程序集与CIL 1. 程序集和CIL: 程序集是由.NET语言的编译器接受源代码文件产生的输出文件,通常分为 exe和dll两类,其中exe包含Main入口方法可以双击执行,dll则需 ...

  7. Spring 从入门到精通 (十六) AOP底层如何创建动态代理类

    关键词:Spring | AOP | 创建代理类 | 底层 本专栏通过理论和实践相结合,系统学习框架核心思想及简单原理,原创不易,如果觉得文章对你有帮助,点赞收藏支持博主 ✨ 目录 一.创建对象三要素 ...

  8. Java动态代理类使用

    Java动态代理类使用 Java动态代理类位于Java.lang.reflect包下,一般主要涉及到以下两个类: 一.Interface InvocationHandler:该接口中仅定义了一个方法O ...

  9. Java之动态代理类实现日志简单实例

    开心一笑 上课呢,同桌撕了一片纸放嘴里了咬了咬,又吐回了手里. 揉了揉,揉成个球状,然后又把这东西给放嘴里,咽下去了. 我问:你干嘛呢? 这二货幽幽地说:这两天有点感冒,捏个药丸吃. 视频教程 大家好 ...

最新文章

  1. 推荐 33 个 IDEA 最牛配置,写代码太爽了!
  2. 一些实用的注册表封装类
  3. 五大场景深解无服务器架构如何实践?
  4. git解决pre-commit hook failed的问题
  5. SEO优化:网站优化的五大步骤
  6. linux 开机提示 Kernel panic - not syncing: Attempted to kill init! 解决方案
  7. 网站优化之如何筛选更正确高质量的关键词?
  8. PHP连接Mysql数据实现增上改查
  9. #include 中的stat.h文件到底在哪个目录下面?
  10. jdi屏幕斜纹_荣耀V10屏幕有斜纹问题,有人甚至因此退货,真的这么严重?
  11. python将list转为数组_python如何将list中的字符转为数字
  12. ak和sk怎么认证 海康威视_JWT和HMAC(AK/SK)认证方式使用场景
  13. Erlang进程堆垃圾回收机制
  14. 我的实用设计模式之关于Policy-based design
  15. 将pdf文件缩小的最快速方法,pdf文件变小的方法
  16. MIS--信息管理系统
  17. Android锁屏勒索病毒分析(4)秒抢红包
  18. Android6.0运行时权限处理
  19. 成都java培训好习惯的养成
  20. 俄罗斯DST收购ICQ;苹果收购芯片提供商(每日关注20100428)

热门文章

  1. 永远不要在代码中使用「User」这个单词!
  2. 2018年Spring Cloud中国社区技术沙龙-成都站
  3. 顶尖学者加盟!两所C9高校,获强援!
  4. CEO亲自写代码登上热榜,从零开始打造“裸金属”树莓派
  5. 施一公:年轻人不要花时间去拉关系,尽全力做研究,以实力取胜!
  6. 医学顶刊BMJ打脸谷歌:AI取代医生检测乳腺癌还远着呢
  7. 人人都能看懂的EM算法推导
  8. 输入示例,自动生成代码:TensorFlow官方工具TF-Coder已开源
  9. 2020年最火本科生专业来了!180所高校新增人工智能专业
  10. Git 高级用法,喜欢就拿去用!