您需要来一份82年的代理吗?No.12

上一篇大家又说我放水了。这样说我很伤心的啵。今天跟大家分享一下代理模式以及JAVA中的代理模式。

代理模式有什么用呢?我总结的一点就是,让别人代理安全一点。

现实生活中其实也有很多代理。

比如保险代理,你不需要去关注太多的细节,只需要做决策是否需要买保险,至于买那种保险适合你,需要哪些资料,都通过代理去帮你管理,虽然要出一点钱,不会浪费自己的时间。

又比如保安,由保安去代理帮你识别出入口的安全问题,自己只需要管理公司内部的事宜。

又比如网络服务供应商代理,由代理去帮你决定使用什么路由决策,使用哪些网线哪些网段,自己只需要保证跟网络供应商的链接是通的。

代理总是要有额外的开销的,这点是毋庸置疑的,无论在生活中还是软件层面中。生活中需要付出额外的时间去寻找代理,额外的金钱去雇佣代理。软件中需要写额外的代码去设计这种代理,或者需要消耗额外的时间去进行代理的调用。

但就一个字,值!

代理模式按用途来分,有以下这么几种:

(1) 远程代理(Remote Proxy)

(2) 虚拟代理(Virtual Proxy)

(3) 保护代理(Protect Proxy)

(4) 缓冲代理(Cache Proxy)

(5) 智能引用代理(Smart Reference Proxy)

远程代理(Remote Proxy)

远程代理就类似网络服务供应商,用来与外面进行通信,内部所有的通信都通过这一个或者几个代理进行,比较方便,也比较安全。有啥事我就谁也不找我就找你了,我就赖在这里不走了哼。

虚拟代理(Virtual Proxy)

怎么说呢,人要脸树要皮,在每加载完成的时候也不希望别人看到苍茫的天涯吧?这是你的爱吗?虚拟机代理就是在大文件加载的时候,先给别人看看,诶你别说,我这里是有料的喔,毋问题你等等就能看到主菜了。

保护代理(Protect Proxy)

保安,控制系统输入输出的安全性。

缓冲缓存代理(Buffer and Cache Proxy)

哎呀呀,这里很多人就不知道缓冲和缓存的区别了,有很多设计中都习惯使用缓冲缓存一起设计的模式,但是缓冲跟缓存是有区别的。

缓冲是什么呢?是避震器,是用来缓解大压力用的,作为一个过渡的区域。缓存又是啥玩意?缓存就是一个方便存储的box,我们用随身的管家来比喻可能好一点,方便,快捷。

我 :我要吃鸡。

管家:没问题马上送到。

管家:您的鸡来了。

我 :这一千万帮我存起来。

管家:没问题您的一千万存起来了。

实际过程中,一般都使用一个内存服务器来进行存储,有就直接取,没有就往其他数据源去请求。这个内存服务器即起到了缓冲的作用,也起到了缓存的作用。

智能引用代理(Smart Reference Proxy)

嘛现在大家能想到的都是这类了。就是在动作的之前之后再做一些动作。比如吃饭这个事情,我不管,我就是要有人帮我记录一下吃饭时间,帮我洗手,帮我试一下菜是不是有毒。(行行行你是皇帝)饭后我不管,也要有人帮我记录吃饭时间,帮我擦嘴帮我洗手帮我刷牙。

按实现方式来说,有这么几种(前方高能我要开始贴大片代码了):

(1) 聚合

(2) 继承

(3) JDK动态代理

(4) CGLIB动态代理

假设我们定义了Loan这么一个接口,实现了LoanCenter这么一个贷款中心。

public interface Loan {
    void loan(Integer loan);
}
public class LoanCenter implements Loan{
    public void loan(Integer loan) {
        Printer.println("loan "+loan+" is handled");
    }
}

为了各种原因,要怎么对这个贷款中心设置代理呢?

聚合

把原始类当成一个成员变量,去前后添油加醋。

public class AggregationLoanProxy extends LoanCenter{
    private Loan loanCenter;
    AggregationLoanProxy(Loan loan){
        this.loanCenter = loan;
    }

    public void loan(Integer loan) {
        Printer.println("Aggregation loan start");
        loanCenter.loan(loan);
        Printer.println("Aggregation loan end");
    }
}

继承

写一个子类去继承LoanCenter,添油加醋,然后调用super方法。

public class InheritLoanProxy extends LoanCenter{
    @Override
    public void loan(Integer loan) {
        Printer.println("inherit loan start");
        super.loan(loan);
        Printer.println("inherit loan start");
    }
}

JDK动态代理

JDK利用反射生成一个子类,添油加醋完去调用接口中的方法。

public class JDKProxy {
    public static Object getProxy(final Object target){
        return Proxy.newProxyInstance(target.getClass().getClassLoader(), target.getClass().getInterfaces(), new InvocationHandler() {
            public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
                Printer.println("jdk loan start");
                Object result = method.invoke(target,args);
                Printer.println("jdk loan end");
                return result;
            }
        });
    }
}

CGLIB动态代理

CGLib利用反射生成一个子类,添油加醋完去调用子类中的目标方法。

public class CGLibProxy implements MethodInterceptor {
    private Object target;
    public Object intercept(Object o, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {
        Printer.println("cglib loan start");
        Object result = methodProxy.invokeSuper(o , args);
        Printer.println("cglib loan end");
        return result;
    }

    public Object getInstance(Object target) {
        this.target = target;
        Enhancer enhancer = new Enhancer();
        enhancer.setSuperclass(this.target.getClass());
        // 回调方法
        enhancer.setCallback(this);
        // 创建代理对象
        return enhancer.create();
    }
}

JDK动态代理和CGLib动态代理都是动态生成子类的方式去进行代理。

什么?你问我JDK和CGLib有什么差别?JDK动态代理只能基于接口,而CGLib是基于方法,所以CGLib的普适性比较高。但是CGLib的开销比JDK动态代理高,性能较差。

还有问题?Spring里面的AOP用的哪个?两个都有使用到,Spring见机行事按需调用。

有小伙伴想知道JDK动态代理和CGLib更深层次的原理吗?私聊我啊万一我下次说说看呢。

原文发布于微信公众号 - 一名叫大蕉的程序员(DaBananaTalk)

原文发表时间:2017-05-05

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏nnngu

记录某公司(简称SMKJ) 的一次面试

昨天去了一家公司面试 Java 开发岗位,这篇文章主要是做一个面试的记录以及总结。

5979
来自专栏Kirito的技术分享

打开orika的正确方式

缘起 架构分层 开发分布式的项目时,DO持久化对象和DTO传输对象的转换是不可避免的。集中式项目中,DO-DAO-SERVICE-WEB的分层再寻常不过,但分布...

67211
来自专栏牛客网

知识总结:设计模式总结(C++和Python实现)前言案例实现 创建型模式 结构型模式行为型模式对比总结

前言 GoF的23种设计模式,包括创建型、结构型和行为型,其涵盖了面向对象思想的精髓以及诸多细节。本文结合《设计模式》和《大话设计模式》,并用C++和Pytho...

6188
来自专栏从流域到海域

《笨办法学Python》 第8课手记

《笨办法学Python》 第8课手记 第八课没有新内容,作者在常见问题解答里面说得很清楚,建议每一课的常见问题解答都要仔细阅读。 代码如下: formatter...

20410
来自专栏Android 研究

Android插件化基础1-----加载SD上APK

大致的意思是: ClassLoader 是一个负责加载classes的对象,ClassLoader类是一个抽象类,需要给出类的二进制名称,ClassLoader...

2493
来自专栏企鹅号快讯

Bruce.Wang-记一次对JS木马分析

0×00 前言 随着 javascript 这类脚本语言的快速发展,它能展现的内容越来越多样,所运用的功能变得越来越强大。当运用在 web 开发上时,增加了更多...

2596
来自专栏小灰灰

基于ForkJoin构建一个简单易用的并发组件

基于ForkJoin构建一个简单易用的并发组件 在实际的业务开发中,需要用到并发编程的知识,实际使用线程池来异步执行任务的场景并不是特别多,而且一般真的遇到了需...

4008
来自专栏Java技术栈

JDK 5 ~ 10 新特性倾情整理!

1321
来自专栏程序员与猫

Go Code Review Comments 译文(截止2018年7月27日)

持续更新中… 原文最新链接 https://github.com/golang/go/wiki/CodeReviewComments/5a40ba36d38...

1643
来自专栏猿人谷

设计模式分类

     在《设计模式》这本书中列举并描述了23种设计模式,分为创建型模式、结构型模式和行为型模式。另外,近来这一清单又增加了一些类别,例如,并发型模式、线程池...

2085

扫码关注云+社区

领取腾讯云代金券