动态代理

  最近整理一些算法的东西,想着得要给每个算法写至少一个测试,要有方法计算运行时间,然后写着写着发现好多功能虽然封装了一边,但是还是有很多代码免不了要重用。比如说算时间的方法,总要在算法方法执行前加行代码,在算法执行后加一行代码。
  忽然就想到了我特么可以用代理类处理啊,这特么不是和打log一样么,所以这里就简单记录一下JDK的动态代理和cglib的动态代理的使用形式。并且这里有个不错的博客:代理模式原理及实例讲解 (小声比比,IBM的博客感觉质量都很高,但是他家的文档就有点emmmmmmmm)

1:JDK动态代理。

首先JDK提供的动态代理需要提供接口以及实现该接口的实现类。

public interface GeneralSort {
    void sort(Integer[] a);
}
public class QuickSort implements GeneralSort{
    //测试
    public static void main(String[] args) {
        GeneralSort generalSortProxy = (GeneralSort)new DynamicProxyTimeCaculateHandler(new QuickSort()).bind();
        generalSortProxy.sort(new int[]{5,4,3,2,1});
    }
    //数组实现
    public  void sort(Integer[] a) {
        quickSort(a, 0, a.length - 1);
    }
    private  void quickSort(Integer[] a,Integer low,Integer high) {
        if (high<=low) {
            return;
        }
        Integer partition = partition(a, low, high);
        quickSort(a, low, partition-1);
        quickSort(a, partition+1, high);
    }
    //第二个循环的判断可删除,因为j==lowd的时候必定等于跳出循环。
    private  Integer partition(Integer[] a,Integer low,Integer high) {
        Integer i=low,j=high+1;
        Integer value  = a[low];
        while(true) {
            while(a[++i]<value) {
                if (i==high) {
                    break;
                }
            }
            while(a[--j]>value) {
                if (j==low) {
                    break;
                }
            }
            if (i>=j) {
                break;
            }
            Integer temp = a[i];
            a[i] = a[j];
            a[j] =temp;
        }
        Integer temp = a[low];
        a[low] = a[j];
        a[j] = temp;
        return j;
    }
}

然后创建一个实现InvocationHandler的代理类。

public class DynamicProxyTimeCaculateHandler implements InvocationHandler {
    //这个就是需要被代理的类了
    private Object object;
    
    public DynamicProxyTimeCaculateHandler(final Object object) {
        this.object = object;   
    }
     //这里的method是接口定义的方法,args是方法传入的参数
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        Integer[] arg = (Integer[])args[0];
        for(Integer i:arg) {
            System.out.print(i+" ");
        }System.out.println();
        Long begainTime = System.nanoTime();
        Object result = method.invoke(object, args);
        Long endTime = System.nanoTime();
        for(Integer i:arg) {
            System.out.print(i+" ");
        }System.out.println();
        System.out.println("运行时间 :"+(endTime-begainTime)+"ns");
        return result;
    }
    //将需要被代理的类进行绑定并且返回代理类
    public Object bind() {
        return (GeneralSort) Proxy.newProxyInstance(object.getClass().getClassLoader(),
                object.getClass().getInterfaces(),
                this);
    }
}

最后执行只需要运行QuickSort中main方法即可。

2.cglib动态代理

cglib的动态代理不需要被代理的类实现某些接口,原理是对指定的类生成一个子类,并覆盖其中方法实现代理,所以final修饰的类是无法被代理的。拦截器定义如下

public class TimeCaculateInterceptor implements MethodInterceptor {

    @Override
    public Object intercept(Object arg0, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {
        printArgs(args);
        Long begainTime = System.nanoTime();
        Object proxyObject = methodProxy.invokeSuper(arg0, args);
        Long endTime = System.nanoTime();
        System.out.println("运行时间 :"+(endTime-begainTime)+"ns");
        return proxyObject;
    }

    private void printArgs(Object[] args) {
        Class pa = args[0].getClass();

        if (pa.isArray()) {
            printrArray((Integer[]) args[0]);
        }
        if (args[0] instanceof ListNode) {
            printListNode((ListNode) args[0]);
        }
    }
}

拦截器的使用如下,这里吐槽一下java,用反射获得的永远是Object类型的实例,意味着每次都需要向下强制转换,就不能一步到位太特么蠢了。老子可终于知道为什么之前用这用那儿的框架的方法一个个要强制装换...

public class ProxyUtil {
    public static Object excuteListSort(String className) throws Exception {
        Class<?> class1 = Class.forName(className);
        TimeCaculateInterceptor timeCaculateInterceptor = new TimeCaculateInterceptor();
        Enhancer enhancer = new Enhancer();
        enhancer.setSuperclass(class1);
        enhancer.setCallback(timeCaculateInterceptor);
        return enhancer.create();
    }
}
public class QuickSort implements GeneralSort{
    private static String className = QuickSort.class.getName();
    
    public static void main(String[] args) {
    
        try {
        QuickSort quickSort = (QuickSort)ProxyUtil.excuteListSort(className);
        printListNode(quickSort.sortList(generateRandomListNode(10)));
        } catch (Exception e) {
        }
        
    }
}

 单纯的只看两个的使用形式,感觉JDK的动态代理就不够灵活,cglib明显好得多。然后就是cglib局限在于不能用于final类,并且效率低于JDK的动态代理。
 最后这里有个疑问,这些代理类都是需要强制转换的,那么像Spring的aspectj这样,如果我不想将类型信息作为hardcode写死,就如上述的ProxyUtil.excuteListSort中我想直接返回一个QuickSort,那么我要如何动态的生成一个能执行的代理类呢...



所有有关的代码都在github

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,657评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,662评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,143评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,732评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,837评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,036评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,126评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,868评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,315评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,641评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,773评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,470评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,126评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,859评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,095评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,584评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,676评论 2 351

推荐阅读更多精彩内容