面试问题:Java里的代理设计模式(Proxy Design Pattern)一共有几种实现方式?这个题目很像孔乙己问“茴香豆的茴字有哪几种写法?”

所谓代理模式,是指客户端(Client)并不直接调用实际的对象(下图右下角的RealSubject),而是通过调用代理(Proxy),来间接的调用实际的对象。

代理模式的使用场合,一般是由于客户端不想直接访问实际对象,或者访问实际的对象存在技术上的障碍,因而通过代理对象作为桥梁,来完成间接访问。

实现方式一:静态代理

开发一个接口IDeveloper,该接口包含一个方法writeCode,写代码。

public interface IDeveloper {public void writeCode();}

创建一个Developer类,实现该接口。

public class Developer implements IDeveloper{private String name;public Developer(String name){this.name = name;}@Overridepublic 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的工作,对着陌生的代码一脸问号。经全组讨论决定,每个开发人员写代码时,必须同步更新文档。

为了强迫每个程序员在开发时记着写文档,而又不影响大家写代码这个动作本身, 我们不修改原来的Developer类,而是创建了一个新的类,同样实现IDeveloper接口。这个新类DeveloperProxy内部维护了一个成员变量,指向原始的IDeveloper实例:

public class DeveloperProxy implements IDeveloper{private IDeveloper developer;public DeveloperProxy(IDeveloper developer){this.developer = developer;}@Overridepublic void writeCode() {System.out.println("Write documentation...");this.developer.writeCode();}
}

这个代理类实现的writeCode方法里,在调用实际程序员writeCode方法之前,加上一个写文档的调用,这样就确保了程序员写代码时都伴随着文档更新。

测试代码:

静态代理方式的优点

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;}@Overridepublic 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;}@Overridepublic void doTesting() {System.out.println("Tester is preparing test documentation...");tester.doTesting();}
}

正是因为有了静态代码方式的这个缺点,才诞生了Java的动态代理实现方式。

Java动态代理实现方式一:InvocationHandler

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);}@Overridepublic 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动态代理的方式去代理它的行为。

Java动态代理实现方式二:CGLIB

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() {@Overridepublic 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实现Java动态代理的局限性

如果我们了解了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的原创技术文章,请关注公众号"汪子熙"或者扫描下面二维码:

Java代理设计模式(Proxy)的四种具体实现:静态代理和动态代理相关推荐

  1. proxy connect abort处理方法_Java代理设计模式(Proxy)的几种具体实现

    Proxy是一种结构设计模型,主要解决对象直接访问带来的问题,代理又分为静态代理和动态代理(JDK代理.CGLIB代理. 静态代理:又程序创建的代理类,或者特定的工具类,在平时开发中经常用到这种代理模 ...

  2. Java代理设计模式(Proxy)的具体实现:静态代理和动态代理

    Java代理设计模式(Proxy)的具体实现:静态代理和动态代理 实现方式一:静态代理 静态代理方式的优点 静态代理方式的缺点 Java动态代理实现方式一:InvocationHandler Java ...

  3. Java反射之创建对象的四种方式

    Java反射之创建对象的四种方式 1.使用new关键字 2.使用Java反射机制,反射构造器 3.使用克隆方式创建对象Cloneable 4.使用序列化Serializable 1.使用Java反射机 ...

  4. java读取XML文件的四种方式

    java读取XML文件的四种方式 Xml代码 <?xml version="1.0" encoding="GB2312"?> <RESULT& ...

  5. JAVA中集合输出的四种方式

    在JAVA中Collection输出有四种方式,分别如下: 一) Iterator输出. 该方式适用于Collection的所有子类. public class Hello {public stati ...

  6. 【零基础学Java】—final关键字与四种用法(二十九)

    [零基础学Java]-final关键字与四种用法(二十九) 一.final关键字 final关键字代表最终.不可改变的 常见的四种用法: 可以用来修饰一个类 可以用来修饰一个方法 可以用来修饰一个局部 ...

  7. java语言变量分为_在Java语言中变量分为四种,分别是___________________________________________。_学小易找答案...

    [填空题]One day, at the registrar's office of a college, I noticed how parents are behaving with their ...

  8. java中Map遍历的四种方式

    java中Map遍历的四种方式 在java中所有的map都实现了Map接口,因此所有的Map(如HashMap, TreeMap, LinkedHashMap, Hashtable等)都可以用以下的方 ...

  9. [转载]java抽取word,pdf的四种武器

    java抽取word,pdf的四种武器 很多人用java进行文档操作时经常会遇到一个问题,就是如何获得word,excel,pdf等文档的内容? 我研究了一下,在这里总结一下抽取word,pdf的几种 ...

最新文章

  1. 爬虫之xpath语法-节点修饰语法
  2. kinect c++
  3. hbuilder ios 打包失败,无法导入p12证书的解决方案
  4. Java学习笔记(13)
  5. python测试开发面试题_python测试开发面试之深浅拷贝
  6. TomCat JDK环境变量
  7. 字符设备驱动笔记(二)
  8. 卸任四家锤子公司法定代表人后:罗永浩退出聊天宝股东行列
  9. hbase 监控指标项
  10. JetBrains产品教育版申请
  11. LINUX系统使用锐捷客户端认证校园网(华中科技大学)
  12. W5500EVB从网络上获取标准时间
  13. c语言程序设计 点菜系统,C语言点餐系统
  14. 黑马程序员之Web前端全栈 · 阶段一 前端开发基础 (1)
  15. word里画的流程图怎么全选_怎么用word画流程图
  16. 2022年京东618店庆活动优惠力度怎么样?
  17. pandas之链式索引问题(chained indexing)
  18. linux 关机和重启命令
  19. Docker 大热,还不了解 Dockerfile 你就OUT啦~
  20. 销售需要具备的8种能力

热门文章

  1. 文献记录(part68)--K- 近邻分类器鲁棒性验证:从约束放松法到随机平滑法
  2. 图卷积神经网络(part3)--三个经典谱域图卷积模型
  3. 聚类(part4)--多源数据聚类算法
  4. 讲讲大厂面试必考的假设检验
  5. 讲几种Python包的安装方式
  6. 给windows 98 客户虚拟机安装VMWare tools
  7. Angular Jasmine单元测试用例spec.ts的加载逻辑
  8. Angular的Zone-Evergreen在SAP Spartacus中的应用
  9. SAP Spartacus delivery mode页面Cannot find control with的错误消息
  10. Angular单元测试如何只执行指定的测试用例,提高测试速度