Java动态代理之InvocationHandler最简单的入门教程

Java动态代理之InvocationHandler最简单的入门教程网上关于 Java 的动态代理 Proxy 和 InvocationHa 这些概念有讲解得非常高深的文章 其实这些概念没有那么复杂 现在咱们通过一个最简单的例子认识什么是 InvocationHa 值得一提的是 InvocationHa 在 Spring 框架实现中被广泛使用 这意味着我们吃透了 InvocationHa 就为将来的 Spring 源码学习打下一个坚实的基础

网上关于Java的动态代理,Proxy和InvocationHandler这些概念有讲解得非常高深的文章。其实这些概念没有那么复杂。现在咱们通过一个最简单的例子认识什么是InvocationHandler。值得一提的是,InvocationHandler在Spring框架实现中被广泛使用,这意味着我们吃透了InvocationHandler,就为将来的Spring源码学习打下一个坚实的基础。

Java动态代理之InvocationHandler最简单的入门教程

开发一个接口,包含两个方法,可以向指定的人问候“你好”或者“再见”。

public interface IHello { 
    void sayHello(String name); void sayGoogBye(String name); } 

创建一个简单的类,实现这个IHello接口。

public class Helloimplements implements IHello { 
    @Override public void sayHello(String name) { 
    System.out.println("Hello " + name); } @Override public void sayGoogBye(String name) { 
    System.out.println(name+" GoodBye!"); } } 

消费这个实现类,迄今为止没什么特别的。

现在假设我们接到了这个需求:老板要求在该实现类每次问候某人时,必须把问候的细节记录到日志文件里。为了简单起见,我们在问候前打印下面的一行语句来模拟日志记录的动作。

System.out.println(“问候之前的日志记录…”);

您也许会说,这还不简单?直接修改Helloimplements的对应方法,把这行日志插入到对应方法即可。

Java动态代理之InvocationHandler最简单的入门教程

然而,老板的要求是:不允许你修改原来的Helloimplements类。在现实场景中,Helloimplements可能是第三方的jar包提供的,我们没有办法修改代码。

Java动态代理之InvocationHandler最简单的入门教程

您也许会说,我们可以用设计模式里的代理模式,即创建一个新的Java类作为代理类,同样实现IHello接口,然后将Helloimplements类的实例传入代理类。我们虽然被要求不允许修改Helloimplements的代码,但是可以把日志记录代码写在代理类里。完整代码如下:

public class StaticProxy implements IHello { 
    private IHello iHello; public void setImpl(IHello impl){ 
    this.iHello = impl; } @Override public void sayHello(String name) { 
    System.out.println("问候之前的日志记录..."); iHello.sayHello(name); } @Override public void sayGoogBye(String name) { 
    System.out.println("问候之前的日志记录..."); iHello.sayGoogBye(name); } static public void main(String[] arg) { 
    Helloimplements hello = new Helloimplements(); StaticProxy proxy = new StaticProxy(); proxy.setImpl(hello); proxy.sayHello("Jerry"); } } 

这种做法能够实现需求:

Java动态代理之InvocationHandler最简单的入门教程

下面我们再看如何用InvocationHandler实现同样的效果。

InvocationHandler是一个JDK提供的标准接口。看下面的代码: import java.lang.reflect.InvocationHandler; import java.lang.reflect.Method; import java.lang.reflect.Proxy; public class DynaProxyHello implements InvocationHandler { 
    private Object delegate; public Object bind(Object delegate) { 
    this.delegate = delegate; return Proxy.newProxyInstance( this.delegate.getClass().getClassLoader(), this.delegate .getClass().getInterfaces(), this); } public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { 
    Object result = null; try { 
    System.out.println("问候之前的日志记录..."); // JVM通过这条语句执行原来的方法(反射机制) result = method.invoke(this.delegate, args); } catch (Exception e) { 
    e.printStackTrace(); } return result; } 

上面代码里的bind方法很想我之前代理类StaticProxy的setImpl方法,只不过这个bind方法的输入参数类型更加通用。日志记录的代码写在方法invoke里。

看看如何使用:

static public void main(String[] arg) { 
    DynaProxyHello helloproxy = new DynaProxyHello(); Helloimplements hello = new Helloimplements(); IHello ihello = (IHello) helloproxy.bind(hello); ihello.sayHello("Jerry"); } } 

执行效果和StaticProxy那种解决方案完全一致。

咱们先来调试一下。当bind方法执行时,方法Proxy.newProxyInstance被调用,Helloimplements类的实例被传入。

Java动态代理之InvocationHandler最简单的入门教程

我们在调试器里观察IHello ihello = (IHello) helloproxy.bind(hello)这行语句返回的ihello变量。虽然它的静态类型是IHello,但请注意,在调试器里观察它的实际类型,并不是Helloimplements的实例,而是JVM给我们加过工的,包含了我们在invoke方法里手写的那行日志记录代码。这个ihello类型为$Proxy0。

Java动态代理之InvocationHandler最简单的入门教程

当这个被JVM加过工的变量的sayHello方法被调用时,JVM自动将调用转交到DynaProxyHello.invoke去:

Java动态代理之InvocationHandler最简单的入门教程

于是,在invoke方法里,我们手写的日志记录代码被执行,然后通过Java反射执行原始的sayHello代码。

有的朋友可能会问,你这个InvocationHandler看起来比静态代理StaticProxy还复杂啊?有什么好处?

假设老板的需求又变了,在调用问候和说再见的方法里,要使用不同的日志记录策略。

看看用InvocationHandler如何优雅实现吧:

Java动态代理之InvocationHandler最简单的入门教程

Java动态代理之InvocationHandler最简单的入门教程

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/228212.html原文链接:https://javaforall.net

(0)
上一篇 2026年3月16日 下午7:33
下一篇 2026年3月16日 下午7:33


相关推荐

  • postfix邮箱_用post方式发送文件

    postfix邮箱_用post方式发送文件一、首先关闭防火墙[root@localhost~]#systemctlstopfirewalld[root@localhost~]#setenforce0[root@localhost~]#getenforcePermissive二、搭建postfix[root@localhost~]#yuminstallpostfix三、配置postfix[root@local…

    2026年2月21日
    4
  • Mybatis开启驼峰命名,作用

    Mybatis开启驼峰命名,作用在 Mybatis 的全局配置文件 mybatis config xml 中加入 configuratio settings settingname mapUnderscor value true settings configuratio ssm

    2026年3月19日
    2
  • JS判断客户端是否已安装ActiveX控件「建议收藏」

    JS判断客户端是否已安装ActiveX控件「建议收藏」if(document.all.player.object==null){alert(“请先安装播放器控件!”)}

    2022年5月15日
    36
  • pycharm好用的主题_pycharm关联python

    pycharm好用的主题_pycharm关联python所谓工欲善其事,必先利其器;在我们日常开发中,长时间编码从眼睛上心里承受压力上有个好的视觉感觉是很加分的以下是我个人十分喜欢的pycharm主题风格,包含整体风格/字体/背景颜色/背景图片;其设置如下:01主题:pycharm的File->Settings->Plugins,搜索MaterialThemeUI并安装,安装之后进行restart02字体:File…

    2022年8月26日
    12
  • Ubuntu 查看磁盘空间大小命令

    Ubuntu 查看磁盘空间大小命令http blog sina com cn s blog 6432901c0100 html nbsp nbsp Df 命令是 linux 系统以磁盘分区为单位查看文件系统 可以加上参数查看磁盘剩余空间信息 命令格式 df hl 显示格式为 文件系统 nbsp nbsp nbsp nbsp nbsp nbsp nbsp nbsp nbsp nbsp nbsp nbsp nbsp 容量已用可用已用 挂载点 Filesystem nbsp nbsp nbsp nbsp nbsp nbsp nbsp nbsp nbsp nbsp nbsp SizeUsedAva

    2026年3月17日
    2
  • 简单介绍BASE64Encoder的使用

    简单介绍BASE64Encoder的使用BASE64Encoder其实是在jkd中的,但是默认不开放,在API中也是找不到的所以先看看怎么将其导入:右击项目–buildpath–>>configurebuildpath–>>双击Accessrules–>>edit–>>edit–>>修改为accessible,RulePatter…

    2022年6月15日
    289

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号