秒懂Java之方法句柄(MethodHandle)

2023-12-12 20:48

本文主要是介绍秒懂Java之方法句柄(MethodHandle),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

【版权申明】非商业目的注明出处可自由转载
博文地址:https://blog.csdn.net/ShuSheng0007/article/details/107066856
出自:shusheng007

文章目录

  • 概述
  • 关键概念
  • 如何使用
    • 创建Lookup
    • 创建MethodType
    • 创建MethodHandle
    • 调用MethodHandle
  • 实际使用
    • 访问构造函数
    • 访问非private实例方法
    • 访问private实例方法
    • 访问非private类方法
    • 访问非private属性
    • 访问private属性
    • 增强MethodHandle
  • 总结
  • 抒情

相关文章:
秒懂Java之反射

概述

众所周知,Java从最初发布时就支持反射,通过反射可以在运行时获取类型信息,但其有个缺点就是执行速度较慢。于是从Java 7开始提供了另一套API MethodHandle 。其与反射的作用类似,可以在运行时访问类型信息,但是据说其执行效率比反射更高,也被称为Java的 现代化反射

官方对其定义如下:

A method handle is a typed, directly executable reference to an underlying method, constructor, field, or similar low-level operation, with optional transformations of arguments or return values.

在《深入理解Java虚拟机》第三版中,作者也提到了MethodHandle, 但作者更多是从JVM的层面理解它,认为其主要目的是为JVM设计的一套API,以支持其他JVM语言的反射能力,例如Groovy 、Scale、Kotlin 等。

本文主要从Java编程语言的角度来看一下如何使用这套API,至于其运行效率是不是真的比反射高,以及高多少都不会涉及,有兴趣的可以自行研究。

关键概念

  • Lookup
    MethodHandle 的创建工厂,通过它可以创建MethodHandle,值得注意的是检查工作是在创建时处理的,而不是在调用时处理。

  • MethodType
    顾名思义,就是代表方法的签名。一个方法的返回值类型是什么,有几个参数,每个参数的类型什么?

  • MethodHandle
    方法句柄,通过它我们就可以动态访问类型信息了。

如何使用

当理解了上面几个关键概念后使用起来就比较简单了,总的来说只需要4步:

  1. 创建Lookup
  2. 创建MethodType
  3. 基于Lookup与MethodType获得MethodHandle
  4. 调用MethodHandle

那我们接下来就按照上面4个步骤通过方法句柄来访问一下某个类里面的方法以及属性等。

首先提供一个目标类

public class HandleTarget {private String name = "hello world";public HandleTarget() {}public HandleTarget(String name) {this.name = name;}public void connectName(String name) {this.name = this.name + " " + name;}public String getName() {return name;}private void learnPrograming(String lang) {System.out.println(String.format("I am learning %s ", lang));}public static String declaration(String author) {return author + ": " + "吾生也有涯,而知也无涯。以有涯随无涯,殆己";}@Overridepublic String toString() {return "HandleTarget{" +"name='" + name + '\'' +'}';}
}

这个类里面有两个构造函数(一个无参,一个有参),一个private Field, 两个public实例方法,一个public static方法以及一个private实例方法。接下来我们就具体看一下如何访问这些元素。

创建Lookup

使用如下代码创建一个lookup,以这种方式得到的lookup很强大,凡是调用类支持的字节码操作,它都支持。

 MethodHandles.Lookup lookup = MethodHandles.lookup();

我们还可以使用如下代码创建,但是以此种方式创建的lookup能力是受限的,其只能访问类中public的成员。

MethodHandles.Lookup publicLookup=MethodHandles.publicLookup();

创建MethodType

MethodType使用其静态方法创建

public static   MethodType methodType(Class<?> rtype, Class<?>[] ptypes)

第一个参数是方法的返回类型,第二参数是方法的入参

其有很多非常方便的重载,基本满足了一般的使用场景

创建MethodHandle

主要通过lookup里面的方法来寻找

  • 创建构造函数MethodHandle
public MethodHandle findConstructor(Class<?> refc, MethodType type) 

refc: 要检索的类
type: 对应的构造函数的MethodType

  • 创建实例方法MethodHandle
 public MethodHandle findVirtual(Class<?> refc, String name, MethodType type)

name: 方法名称

  • 创建类方法的MethodHandle
public   MethodHandle findStatic(Class<?> refc, String name, MethodType type)
  • 创建非private的Field的访问MethodHandle 。

注意这个不是获取field的javabean Setter方法,与其毫无关系。通过这个setter 方法句柄我们就可以访问到这个属性了。

public MethodHandle findGetter(Class<?> refc, String name, Class<?> type)

对应的如果要设置此属性的值,使用Setter方法句柄

public MethodHandle findSetter(Class<?> refc, String name, Class<?> type)

调用MethodHandle

使用MethodHandle的invoke家族方法

public final native @PolymorphicSignature Object invoke(Object... args) throws Throwable;
public final native @PolymorphicSignature Object invokeExact(Object... args) throws Throwable;
...

实际使用

首先创建一个lookup

MethodHandles.Lookup lookup = MethodHandles.lookup();

访问构造函数

//无参数构造器
MethodType con1Mt = MethodType.methodType(void.class);
MethodHandle con1Mh = lookup.findConstructor(HandleTarget.class, con1Mt);
Object target1 = con1Mh.invoke();
//有参数构造器
MethodType con2Mt = MethodType.methodType(void.class, String.class);
MethodHandle con2Mh = lookup.findConstructor(HandleTarget.class, con2Mt);
Object target2 = con2Mh.invoke("ErGouWang");

访问非private实例方法

//调用非private实例方法
MethodType getterMt = MethodType.methodType(String.class);
MethodHandle getterMh = lookup.findVirtual(HandleTarget.class, "getName", getterMt);
String name = (String) getterMh.invoke(target2);
System.out.println(name);

访问private实例方法

//访问private方法
Method learnMethod = HandleTarget.class.getDeclaredMethod("learnPrograming", String.class);
learnMethod.setAccessible(true);
MethodHandle learnProMh = lookup.unreflect(learnMethod);
learnProMh.invoke(target1, "Java");

访问非private类方法

//调用静态方法
MethodType decMt = MethodType.methodType(String.class, String.class);
MethodHandle decMh = lookup.findStatic(HandleTarget.class, "declaration", decMt);
String dec = (String) decMh.invoke("庄子");
System.out.println(dec);

访问非private属性

//访问非private属性
MethodHandle nameMh= lookup.findGetter(HandleTarget.class,"name", String.class);
System.out.println((String) nameMh.invoke(con1Mh.invoke()));

访问private属性

//访问private的属性,需要借助反射
Field nameField = HandleTarget.class.getDeclaredField("name");
nameField.setAccessible(true);
MethodHandle nameFromRefMh = lookup.unreflectGetter(nameField);
System.out.println((String) nameFromRefMh.invoke(target1));

增强MethodHandle

//增强MethodHandle
MethodType setterMt = MethodType.methodType(void.class, String.class);
MethodHandle setterMh = lookup.findVirtual(HandleTarget.class, "connectName", setterMt);
MethodHandle bindedSetterMh = setterMh.bindTo(target2);
bindedSetterMh.invoke("love CuiHuaNiu");
System.out.println((String) getterMh.invoke(target2));

当我们创建了"connectName"方法的MethodHandle,可以不立即调用而是将其绑定到某个对象上,这个对象的类型必须是HandleTarget及其子类,那么调用重新获得MethodHandle时,其会调用到新绑定对象里面的那个方法上。

总结

江湖流传着一种说法:使用MethodHandle就像是在用Java来写字节码。这种说法是有一定道理的,因为MethodHandle里的很多操作都对应着相应的字节码。总的来说,其与反射一样,离应用型程序员日常开发比较远,但是在开发框架和和工具包时却会被大量使用。

抒情

人生在世不称意,明朝散发弄扁舟

本文源码地址:ToMasterJava

这篇关于秒懂Java之方法句柄(MethodHandle)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/485894

相关文章

Java 正则表达式URL 匹配与源码全解析

《Java正则表达式URL匹配与源码全解析》在Web应用开发中,我们经常需要对URL进行格式验证,今天我们结合Java的Pattern和Matcher类,深入理解正则表达式在实际应用中... 目录1.正则表达式分解:2. 添加域名匹配 (2)3. 添加路径和查询参数匹配 (3) 4. 最终优化版本5.设计思

Java使用ANTLR4对Lua脚本语法校验详解

《Java使用ANTLR4对Lua脚本语法校验详解》ANTLR是一个强大的解析器生成器,用于读取、处理、执行或翻译结构化文本或二进制文件,下面就跟随小编一起看看Java如何使用ANTLR4对Lua脚本... 目录什么是ANTLR?第一个例子ANTLR4 的工作流程Lua脚本语法校验准备一个Lua Gramm

Java字符串操作技巧之语法、示例与应用场景分析

《Java字符串操作技巧之语法、示例与应用场景分析》在Java算法题和日常开发中,字符串处理是必备的核心技能,本文全面梳理Java中字符串的常用操作语法,结合代码示例、应用场景和避坑指南,可快速掌握字... 目录引言1. 基础操作1.1 创建字符串1.2 获取长度1.3 访问字符2. 字符串处理2.1 子字

Java Optional的使用技巧与最佳实践

《JavaOptional的使用技巧与最佳实践》在Java中,Optional是用于优雅处理null的容器类,其核心目标是显式提醒开发者处理空值场景,避免NullPointerExce... 目录一、Optional 的核心用途二、使用技巧与最佳实践三、常见误区与反模式四、替代方案与扩展五、总结在 Java

基于Java实现回调监听工具类

《基于Java实现回调监听工具类》这篇文章主要为大家详细介绍了如何基于Java实现一个回调监听工具类,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录监听接口类 Listenable实际用法打印结果首先,会用到 函数式接口 Consumer, 通过这个可以解耦回调方法,下面先写一个

使用Java将DOCX文档解析为Markdown文档的代码实现

《使用Java将DOCX文档解析为Markdown文档的代码实现》在现代文档处理中,Markdown(MD)因其简洁的语法和良好的可读性,逐渐成为开发者、技术写作者和内容创作者的首选格式,然而,许多文... 目录引言1. 工具和库介绍2. 安装依赖库3. 使用Apache POI解析DOCX文档4. 将解析

Java字符串处理全解析(String、StringBuilder与StringBuffer)

《Java字符串处理全解析(String、StringBuilder与StringBuffer)》:本文主要介绍Java字符串处理全解析(String、StringBuilder与StringBu... 目录Java字符串处理全解析:String、StringBuilder与StringBuffer一、St

springboot整合阿里云百炼DeepSeek实现sse流式打印的操作方法

《springboot整合阿里云百炼DeepSeek实现sse流式打印的操作方法》:本文主要介绍springboot整合阿里云百炼DeepSeek实现sse流式打印,本文给大家介绍的非常详细,对大... 目录1.开通阿里云百炼,获取到key2.新建SpringBoot项目3.工具类4.启动类5.测试类6.测

Python列表去重的4种核心方法与实战指南详解

《Python列表去重的4种核心方法与实战指南详解》在Python开发中,处理列表数据时经常需要去除重复元素,本文将详细介绍4种最实用的列表去重方法,有需要的小伙伴可以根据自己的需要进行选择... 目录方法1:集合(set)去重法(最快速)方法2:顺序遍历法(保持顺序)方法3:副本删除法(原地修改)方法4:

Spring Boot循环依赖原理、解决方案与最佳实践(全解析)

《SpringBoot循环依赖原理、解决方案与最佳实践(全解析)》循环依赖指两个或多个Bean相互直接或间接引用,形成闭环依赖关系,:本文主要介绍SpringBoot循环依赖原理、解决方案与最... 目录一、循环依赖的本质与危害1.1 什么是循环依赖?1.2 核心危害二、Spring的三级缓存机制2.1 三