广告
返回顶部
首页 > 资讯 > 精选 >如何进行Java代理设计模式的静态代理和动态代理实现
  • 405
分享到

如何进行Java代理设计模式的静态代理和动态代理实现

2023-06-02 22:06:47 405人浏览 薄情痞子
摘要

今天就跟大家聊聊有关如何进行Java代理设计模式的静态代理和动态代理实现,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。面试问题:Java里的代理设计模式一共有几种实现方式?这个题目很

今天就跟大家聊聊有关如何进行Java代理设计模式的静态代理和动态代理实现,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。

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

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

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

如何进行Java代理设计模式的静态代理和动态代理实现

实现方式一:静态代理

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

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的工作,对着陌生的代码一脸问号。经全组讨论决定,每个开发人员写代码时,必须同步更新文档。

为了强迫每个程序员在开发时记着写文档,而又不影响大家写代码这个动作本身, 我们不修改原来的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方法之前,加上一个写文档的调用,这样就确保了程序员写代码时都伴随着文档更新。

测试代码:

如何进行Java代理设计模式的静态代理和动态代理实现

静态代理方式的优点

易于理解和实现

代理类和真实类的关系是编译期静态决定的,和下文马上要介绍的动态代理比较起来,执行时没有任何额外开销。

静态代理方式的缺点

每一个真实类都需要一个创建新的代理类。还是以上述文档更新为例,假设老板对测试工程师也提出了新的要求,让测试工程师每次测出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的动态代理实现方式。

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);    }    @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方法在动态代理类里通过反射的方式进行执行。

测试输出:

如何进行Java代理设计模式的静态代理和动态代理实现

通过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代理设计模式的静态代理和动态代理实现

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() {            @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未实现任何代码,但它也成功被代理了:

如何进行Java代理设计模式的静态代理和动态代理实现

用CGLIB实现Java动态代理的局限性

如果我们了解了CGLIB创建代理类的原理,那么其局限性也就一目了然。我们现在做个实验,将ProductOwner类加上final修饰符,使其不可被继承:

如何进行Java代理设计模式的静态代理和动态代理实现

再次执行测试代码,这次就报错了: Cannot subclass final class XXXX。

所以通过CGLIB成功创建的动态代理,实际是被代理类的一个子类。那么如果被代理类被标记成final,也就无法通过CGLIB去创建动态代理。

Java动态代理实现方式三:通过编译期提供的API动态创建代理类

假设我们确实需要给一个既是final,又未实现任何接口的ProductOwner类创建动态代码。除了InvocationHandler和CGLIB外,我们还有最后一招:

我直接把一个代理类的源代码用字符串拼出来,然后基于这个字符串调用jdk的Compiler(编译期)API,动态的创建一个新的.java文件,然后动态编译这个.java文件,这样也能得到一个新的代理类。

如何进行Java代理设计模式的静态代理和动态代理实现

测试成功:

如何进行Java代理设计模式的静态代理和动态代理实现

我拼好了代码类的源代码,动态创建了代理类的.java文件,能够在Eclipse里打开这个用代码创建的.java文件,

如何进行Java代理设计模式的静态代理和动态代理实现

如何进行Java代理设计模式的静态代理和动态代理实现

下图是如何动态创建ProductPwnerSCProxy.java文件:

如何进行Java代理设计模式的静态代理和动态代理实现

下图是如何用JavaCompiler API动态编译前一步动态创建出的.java文件,生成.class文件:

如何进行Java代理设计模式的静态代理和动态代理实现

下图是如何用类加载器加载编译好的.class文件到内存:

如何进行Java代理设计模式的静态代理和动态代理实现

看完上述内容,你们对如何进行Java代理设计模式的静态代理和动态代理实现有进一步的了解吗?如果还想了解更多知识或者相关内容,请关注编程网精选频道,感谢大家的支持。

--结束END--

本文标题: 如何进行Java代理设计模式的静态代理和动态代理实现

本文链接: https://www.lsjlt.com/news/231500.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • 如何进行Java代理设计模式的静态代理和动态代理实现
    今天就跟大家聊聊有关如何进行Java代理设计模式的静态代理和动态代理实现,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。面试问题:Java里的代理设计模式一共有几种实现方式?这个题目很...
    99+
    2023-06-02
  • java代理模式(静态代理、动态代理、cglib代理)
    目录代理模式静态代理代码接口被代理对象代理对象测试动态代理代码:接口目标对象代理对象测试cglib代理代码:目标对象代理对象测试应用总结代理模式 代理模式(Proxy Pattern...
    99+
    2022-11-12
  • 代理模式:JAVA静态代理和动态代理的实例和实现详解
    目录前言静态代理实现简述创建human接口创建接口实现类创建针对接口实现增强操作的代理代理实现效果动态代理实现简述要点:向上转型创建YoungMan接口创建两个接口实现类创建动态代理...
    99+
    2022-11-12
  • Java的动态代理和静态代理详解
    目录0、代理模式1、静态代理2、 加深理解3、动态代理动态代理的例子总结0、代理模式 为什么要学习代理模式?这是SpringAOP的底层【SpringAOP和SpringMVC】 代...
    99+
    2022-11-13
  • 使用Java怎么实现静态代理和动态代理
    本篇文章为大家展示了使用Java怎么实现静态代理和动态代理,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。代理模式代理模式(Proxy):为其他对象提供一个代理以控制对这个对象的访问。主要解决:在直接...
    99+
    2023-06-14
  • Java中怎么实现静动态代理模式
    这篇文章将为大家详细讲解有关Java中怎么实现静动态代理模式,文章内容质量较高,因此小编分享给大家做个参考,希望大家阅读完这篇文章后对相关知识有一定的了解。1.静态代理模式 使用场合:为其他对象提供一种代理以控制对这个对象的访问。在某些情况...
    99+
    2023-06-17
  • Java静态代理和动态代理的深入讲解
    代理模式 代理模式(Proxy):为其他对象提供一个代理以控制对这个对象的访问。 主要解决:在直接访问对象时带来的问题,比如说:要访问的对象在远程的机器上。在面向对象系统中,有些对...
    99+
    2022-11-12
  • 怎么理解Java和SAP ABAP的静态代理和动态代理
    本篇内容介绍了“怎么理解Java和SAP ABAP的静态代理和动态代理”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!下图是某应用里方法的常规...
    99+
    2023-06-02
  • Java中动态代理和静态代理的示例分析
    这篇文章主要介绍了Java中动态代理和静态代理的示例分析,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。0、代理模式为什么要学习代理模式?这是SpringAOP的底层【Spri...
    99+
    2023-06-29
  • Spring细数两种代理模式之静态代理和动态代理概念及使用
    目录代理模式1、静态代理2、动态代理代理模式 在学习 AOP 之前,我们先来了解下代理模式, 代理模式分为静态代理以及动态代理,属于23中设计模式之一。 AOP 的底层机制就是动态代...
    99+
    2023-02-27
    Spring静态代理与动态代理 Spring静态代理 Spring动态代理
  • 详解Java中的静态代理模式
    目录静态代理模式代理模式静态代理模式的实现要求举例静态代理模式 代理模式 代理人可以替受代理的人完成一些工作,比如:学校与学生之间的关系来说,学生就好比被被代理类,学生只需要负责上课...
    99+
    2022-12-26
    Java静态代理模式 Java静态代理
  • Java实现多线程中的静态代理模式
    前言 代理模式是一种设计模式,提供了对目标对象额外的访问方式,即通过代理对象访问目标对象,这样可以在不修改原目标对象的前提下,提供额外的功能操作,扩展目标对象的功能。 简言之,代理模...
    99+
    2022-11-12
  • Java动态代理模式怎么实现
    这篇文章主要介绍“Java动态代理模式怎么实现”,在日常操作中,相信很多人在Java动态代理模式怎么实现问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Java动态代理模式怎么实现”的疑惑有所帮助!接下来,请跟...
    99+
    2023-06-17
  • java设计模式—静态代理模式(聚合与继承方式对比)
    一、概述目标:要在Tank的move()方法做时间代理及日志代理(可以设想以后还要增加很多代理处理),且代理间的顺序可活更换思路:(1)聚合:代理类聚合了被代理类,且代理类及被代理类都实现了movable接口,则可实现灵活多变,具体看代码(...
    99+
    2023-05-31
    java 静态代理模式 设计模式
  • Java动态代理如何实现
    本篇内容介绍了“Java动态代理如何实现”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!JDK动态代理:利用反射机制生成一个实现代理接口的匿名...
    99+
    2023-06-30
  • Java怎么实现多线程中的静态代理模式
    这篇文章将为大家详细讲解有关Java怎么实现多线程中的静态代理模式,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。前言代理模式是一种设计模式,提供了对目标对象额外的访问方式,即通过代理对象访问目标对象,这样...
    99+
    2023-06-15
  • 老生常谈设计模式之动态代理
    一、动态代理概念动态代理分为JDK动态代理和cglib动态代理两种方式。jdk动态代理是由Java内部的反射机制来实现的,cglib动态代理底层则是借助asm来实现的。总的来说,反射机制在生成类的过程中比较高效,而asm在生成类之后的相关执...
    99+
    2023-05-31
    设计模式 动态代理
  • Java实现动态代理的实例代码
    目录前言静态代理 动态代理 CGLib实现动态代理 总结前言 动态代理在Java中有着广泛的应用,比如Spring AOP、Hibernate数据查询、测试框架的后端mock、RPC...
    99+
    2022-11-12
  • Java如何实现JDK动态代理
    这篇文章主要讲解了“Java如何实现JDK动态代理”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Java如何实现JDK动态代理”吧!概念代理:为控制A对象,而创建出新B对象,由B对象代替执行...
    99+
    2023-07-02
  • Java设计模式之代理模式原理及实现代码分享
    简介Java编程的目标是实现现实不能完成的,优化现实能够完成的,是一种虚拟技术。生活中的方方面面都可以虚拟到代码中。代理模式所讲的就是现实生活中的这么一个概念:中介。代理模式的定义:给某一个对象提供一个代理,并由代理对象控制对原对象的引用。...
    99+
    2023-05-30
    java 设计模式 代理模式
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作