面试问题:Java里的代理设计模式(Proxy Design Pattern)一共有几种实现方式?这个题目很像孔乙己问“茴香豆的茴字有哪几种写法?”java
所谓代理模式,是指客户端(Client)并不直接调用实际的对象(下图右下角的RealSubject),而是经过调用代理(Proxy),来间接的调用实际的对象。git
代理模式的使用场合,通常是因为客户端不想直接访问实际对象,或者访问实际的对象存在技术上的障碍,于是经过代理对象做为桥梁,来完成间接访问。程序员
开发一个接口IDeveloper,该接口包含一个方法writeCode,写代码。github
public interface IDeveloper { public void writeCode(); }
建立一个Developer类,实现该接口。面试
public class Developer implements IDeveloper{ private String name; public Developer(String name){ this.name = name; } @Override public void writeCode() { System.out.println("Developer " + name + " writes code"); } }
测试代码:建立一个Developer实例,名叫Jerry,去写代码!设计模式
public class DeveloperTest { public static void main(String[] args) { IDeveloper jerry = new Developer("Jerry"); jerry.writeCode(); } }
如今问题来了。Jerry的项目经理对Jerry光写代码,而不维护任何的文档很不满。假设哪天Jerry休假去了,其余的程序员来接替Jerry的工做,对着陌生的代码一脸问号。经全组讨论决定,每一个开发人员写代码时,必须同步更新文档。ide
为了强迫每一个程序员在开发时记着写文档,而又不影响你们写代码这个动做自己, 咱们不修改原来的Developer类,而是建立了一个新的类,一样实现IDeveloper接口。这个新类DeveloperProxy内部维护了一个成员变量,指向原始的IDeveloper实例:测试
public class DeveloperProxy implements IDeveloper{ private IDeveloper developer; public DeveloperProxy(IDeveloper developer){ this.developer = developer; } @Override public void writeCode() { System.out.println("Write documentation..."); this.developer.writeCode(); } }
这个代理类实现的writeCode方法里,在调用实际程序员writeCode方法以前,加上一个写文档的调用,这样就确保了程序员写代码时都伴随着文档更新。this
测试代码:设计
1. 易于理解和实现
2. 代理类和真实类的关系是编译期静态决定的,和下文立刻要介绍的动态代理比较起来,执行时没有任何额外开销。
每个真实类都须要一个建立新的代理类。仍是以上述文档更新为例,假设老板对测试工程师也提出了新的要求,让测试工程师每次测出bug时,也要及时更新对应的测试文档。那么采用静态代理的方式,测试工程师的实现类ITester也得建立一个对应的ITesterProxy类。
public interface ITester { public void doTesting(); } Original tester implementation class: public class Tester implements ITester { private String name; public Tester(String name){ this.name = name; } @Override public void doTesting() { System.out.println("Tester " + name + " is testing code"); } } public class TesterProxy implements ITester{ private ITester tester; public TesterProxy(ITester tester){ this.tester = tester; } @Override public void doTesting() { System.out.println("Tester is preparing test documentation..."); tester.doTesting(); } }
正是由于有了静态代码方式的这个缺点,才诞生了Java的动态代理实现方式。
InvocationHandler的原理我曾经专门写文章介绍过:Java动态代理之InvocationHandler最简单的入门教程
经过InvocationHandler, 我能够用一个EnginnerProxy代理类来同时代理Developer和Tester的行为。
public class EnginnerProxy implements InvocationHandler { Object obj; public Object bind(Object obj) { this.obj = obj; return Proxy.newProxyInstance(obj.getClass().getClassLoader(), obj .getClass().getInterfaces(), this); } @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { System.out.println("Enginner writes document"); Object res = method.invoke(obj, args); return res; } }
真实类的writeCode和doTesting方法在动态代理类里经过反射的方式进行执行。
测试输出:
经过InvocationHandler实现动态代理的局限性
假设有个产品经理类(ProductOwner) 没有实现任何接口。
public class ProductOwner { private String name; public ProductOwner(String name){ this.name = name; } public void defineBackLog(){ System.out.println("PO: " + name + " defines Backlog."); } }
咱们仍然采起EnginnerProxy代理类去代理它,编译时不会出错。运行时会发生什么事?
ProductOwner po = new ProductOwner("Ross"); ProductOwner poProxy = (ProductOwner) new EnginnerProxy().bind(po); poProxy.defineBackLog();
运行时报错。因此局限性就是:若是被代理的类未实现任何接口,那么不能采用经过InvocationHandler动态代理的方式去代理它的行为。
CGLIB是一个Java字节码生成库,提供了易用的API对Java字节码进行建立和修改。关于这个开源库的更多细节,请移步至CGLIB在github上的仓库:https://github.com/cglib/cglib
咱们如今尝试用CGLIB来代理以前采用InvocationHandler没有成功代理的ProductOwner类(该类未实现任何接口)。
如今我改成使用CGLIB API来建立代理类:
public class EnginnerCGLibProxy { Object obj; public Object bind(final Object target) { this.obj = target; Enhancer enhancer = new Enhancer(); enhancer.setSuperclass(obj.getClass()); enhancer.setCallback(new MethodInterceptor() { @Override public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable { System.out.println("Enginner 2 writes document"); Object res = method.invoke(target, args); return res; } } ); return enhancer.create(); } }
测试代码:
ProductOwner ross = new ProductOwner("Ross"); ProductOwner rossProxy = (ProductOwner) new EnginnerCGLibProxy().bind(ross); rossProxy.defineBackLog();
尽管ProductOwner未实现任何代码,但它也成功被代理了:
若是咱们了解了CGLIB建立代理类的原理,那么其局限性也就一目了然。咱们如今作个实验,将ProductOwner类加上final修饰符,使其不可被继承:
再次执行测试代码,此次就报错了: Cannot subclass final class XXXX。
因此经过CGLIB成功建立的动态代理,实际是被代理类的一个子类。那么若是被代理类被标记成final,也就没法经过CGLIB去建立动态代理。
Java动态代理实现方式三:经过编译期提供的API动态建立代理类
假设咱们确实须要给一个既是final,又未实现任何接口的ProductOwner类建立动态代码。除了InvocationHandler和CGLIB外,咱们还有最后一招:
我直接把一个代理类的源代码用字符串拼出来,而后基于这个字符串调用JDK的Compiler(编译期)API,动态的建立一个新的.java文件,而后动态编译这个.java文件,这样也能获得一个新的代理类。
测试成功:
我拼好了代码类的源代码,动态建立了代理类的.java文件,可以在Eclipse里打开这个用代码建立的.java文件,
下图是如何动态建立ProductPwnerSCProxy.java文件:
下图是如何用JavaCompiler API动态编译前一步动态建立出的.java文件,生成.class文件:
下图是如何用类加载器加载编译好的.class文件到内存:
若是您想试试这篇文章介绍的这四种代理模式(Proxy Design Pattern), 请参考个人github仓库,所有代码都在上面。感谢阅读。
https://github.com/i042416/JavaTwoPlusTwoEquals5/tree/master/src/proxy
要获取更多Jerry的原创技术文章,请关注公众号"汪子熙"或者扫描下面二维码: