现在的位置: 首页 > 综合 > 正文

代理模式之强制代理模式以及动态代理原理

2018年01月25日 ⁄ 综合 ⁄ 共 9702字 ⁄ 字号 评论关闭

 

1:强制代理模式

         普通代理模式和强制代理模式是代理模式的其中俩个部分。对于这俩个代理模式,我们采用一个类似事情解释下:

在网络上的代理服务器分透明代理和普通代理;对于透明代理来说,字如其意,就是这个代理服务器对用户来说是透明的,用户根本不知道这个服务器的存在。而普通代理则是用户需要自己手动去设置代理服务器的ip,很显然,对于这点上,用户必须知道代理服务器的存在。现在对号入座:代理模式中的强制代理类似网络上的代理服务器的透明代理,代理模式中的普通代理模式就类似网络上的代理服务器的普通代理。

普通代理模式比较常见,在这里就不作多说了,那么在这里就重点来看看强制代理模式。

 

        强制代理模式在设计模式中比较怪怪的,很另类。一般的思维都是通过代理模式找到真实的角色,但是强制代理模式反其道而行之,必须强制通过真实角色去查找到到模式,否则不能访问。换句话说就是有真实角色去管理代理角色。总而言之:就是通过new或者其他方式创建一个真实角色的对象,最后却是会有真实角色返回他自己的代理角色。类似日常生活中:哥们咱想通过给点钱给某某局长,让他出面给咱们半个事情,结果他给我们一张他秘书的名片(理由很简单,这种事情当老大的不会出面),让他们本想通过局长要做的事情,直接通过秘书去做。其实:秘书能做这些,也就是此局长的权势;说白了秘书就是该局长的替身,代理。上图:

 

在接口上设置一个getProxy()方法,真实角色就能可以指定一个自己的代理。除了代理,谁也不能访问。很显然,当官的人都是会让指定的亲信做事,其他人信不过,当然不能胜任此事。上代码:局长抑或秘书至少都是官员

 

 

 

 

真实角色和代理角色都应该实现上面的基类。

 

真实角色,增加一个私有方法,检查是否为自己的代理,如果是自己的代理,才能执行自己的方法。接下来看看代替实现:

 

 

 

 

 下面将演示3中场景:

 

运行结果:

 

场景1:

please user pointed proxy receiveMoney

please user pointed proxy sendMoney

 

场景2:

please user pointed proxy receiveMoney

please user pointed proxy sendMoney

 

场景3:

张局长 are starting to receiveMoney

张局长 are starting to sendMoney

 

        所以从上面的实例中可以得出:强制代理的概念就是必须从真实角色中查找返回其代理角色,不允许直接通过调用真实角色的方法,因为在每次调用真实角色从基类继承的方法时候,该方法都会先去判断是否为代理调用。高层模块只需访问真实角色的getProxy(),就可以完成访问真实角色的所有方法,它根本就不需要产生一个代理处理,代理的管理已经由真实角色自己完成。强制代理模式使得高层模块与真实角色的业务场景耦合度极地;真实角色的修改,在高层模块中完全不用修改。俩者只通过真实角色提供的一个getProxy()接口来交互。在整个实现过程,采用回调思想,在真实角色中调用代理对象的方法,代理对象通过回调来调用真实角色的方法。

 

2:动态代理模式

       先解释下何为动态代理模式,所谓动态代理模式就是在实现阶段毋须关心代理谁,只有在运行阶段才指定代理哪一个对象。通俗点讲就是:自己动手写代理类的方式就是静态代理模式。采用动态代理模式比较为大家所熟知的比如AOP(切面编程),其核心机制就是采用动态代理机制。我们继续使用上面的官员行贿和受贿的例子来解析动态代理模式原理:上图

 

 

 

 

 

在类图中增加一个InvocationHandler接口;其中InvocationHandler是JDK提供的动态代理接口,对被代理的类(即是真实角色的对象,譬如上例中的局长角色)的方法进行代理。看代码:

其中invoke方法是InvocationHandler派生类必须实现的方法,它完成对真实角色(秘书调用局长的方法)的方法的调用。动态代理就是根据被代理的真实角色所属的接口生成所有方法,换而言之:代理对象已经实现接口下(IOfficer)的所有方法。类似静态代理模式:public class OfficerProxy implements IOfficer 也要实现该接口下所有方法。

 


 

 

运行结果:

张局长 are starting to receiveMoney
张局长 are starting to sendMoney

 

粗看很不解,IOfficer proxy是生成的代理类,但是没有看到代理类实现接口的方法。(IOfficer) Proxy.newProxyInstance来生成代理对象,里面到底是如何处理的,我们进去看看Proxy.newProxyInstance这个方法:

 

 

可见关键之处在于Class cl = getProxyClass(loader, interfaces);Proxy利用类装载器和代理类的接口来生成一个实现了接口各个方法的代理类的class对象。我们分析下代码getProxyClass(loader, interfaces)代码

 

 

所以:Proxy.newProxyInstance内部机制为:

 

1:实现接口
2:装载字节流文件到虚拟机
3:实例化,并返回对象

 

类推:代理对象调用方法:Proxy.sendMoney(),他的实现应该类似如何:

通过InvocationHandler的invoke方法类来调用真实角色对应方法

 

 

 

 

 

 

抱歉!评论已关闭.