当前位置:首页 > 科技  > 软件

通过方法引用获取属性名的底层逻辑是什么?

来源: 责编: 时间:2024-04-11 09:04:24 236观看
导读很多小伙伴可能都用过 MyBatis-Plus,这里边我们构造 where 条件的时候,可以直接通过方法引用的方式去指定属性名:LambdaQueryWrapper<Book> qw = new LambdaQueryWrapper<>();qw.eq(Book::getId, 2);List<Book> list = b

很多小伙伴可能都用过 MyBatis-Plus,这里边我们构造 where 条件的时候,可以直接通过方法引用的方式去指定属性名:iBd28资讯网——每日最新资讯28at.com

LambdaQueryWrapper<Book> qw = new LambdaQueryWrapper<>();qw.eq(Book::getId, 2);List<Book> list = bookMapper.selectList(qw);System.out.println("list = " + list);

Book::getId 这就是方法引用,松哥之前也专门写过文章介绍相关内容,这里就不再多说。这里我们就单纯来说说为什么 MP 通过 Book::getId 就可以识别出来这里的属性名。iBd28资讯网——每日最新资讯28at.com

1. 源码分析

这个问题其实好解决,我们顺着 qw.eq 这个方法往下看就可以了,这个方法在执行的过程中几经辗转会来到 getColumnCache 方法中,这个方法就是解析出来属性值的地方。iBd28资讯网——每日最新资讯28at.com

protected ColumnCache getColumnCache(SFunction<T, ?> column) {    LambdaMeta meta = LambdaUtils.extract(column);    String fieldName = PropertyNamer.methodToProperty(meta.getImplMethodName());    Class<?> instantiatedClass = meta.getInstantiatedClass();    tryInitCache(instantiatedClass);    return getColumnCache(fieldName, instantiatedClass);}

首先这里先将我们传入的 Lambda 表达式通过 LambdaUtils.extract 方法解析出来一个 LambdaMeta 对象。iBd28资讯网——每日最新资讯28at.com

public static <T> LambdaMeta extract(SFunction<T, ?> func) {    // 1. IDEA 调试模式下 lambda 表达式是一个代理    if (func instanceof Proxy) {        return new IdeaProxyLambdaMeta((Proxy) func);    }    // 2. 反射读取    try {        Method method = func.getClass().getDeclaredMethod("writeReplace");        method.setAccessible(true);        return new ReflectLambdaMeta((SerializedLambda) method.invoke(func), func.getClass().getClassLoader());    } catch (Throwable e) {        // 3. 反射失败使用序列化的方式读取        return new ShadowLambdaMeta(com.baomidou.mybatisplus.core.toolkit.support.SerializedLambda.extract(func));    }}

这块的重点其实就在反射读取这块,这是从我们传入的 Lambda 中找到了一个名为 writeReplace 的方法,并且通过反射执行了这个方法,然后将执行结果封装为一个 ReflectLambdaMeta 对象返回。iBd28资讯网——每日最新资讯28at.com

接下来回到 getColumnCache 方法中,继续通过 String fieldName = PropertyNamer.methodToProperty(meta.getImplMethodName()); 获取到属性名称。iBd28资讯网——每日最新资讯28at.com

这里有一个 meta.getImplMethodName() 方法,这个方法的拿到的其实就是我们 Lambda 表达式中的方法名,也就是 getId,然后再通过 PropertyNamer.methodToProperty 对这个方法名进行处理,最终拿到属性名:iBd28资讯网——每日最新资讯28at.com

public static String methodToProperty(String name) {  if (name.startsWith("is")) {    name = name.substring(2);  } else if (name.startsWith("get") || name.startsWith("set")) {    name = name.substring(3);  } else {    throw new ReflectionException(        "Error parsing property name '" + name + "'.  Didn't start with 'is', 'get' or 'set'.");  }  if (name.length() == 1 || name.length() > 1 && !Character.isUpperCase(name.charAt(1))) {    name = name.substring(0, 1).toLowerCase(Locale.ENGLISH) + name.substring(1);  }  return name;}

大家看到,这个解析的过程其实就是把方法名的前缀 get/set/is 这些去掉,然后剩余的字符串首字母小写之后返回。iBd28资讯网——每日最新资讯28at.com

这就是我们传入 Book::getId,最终能够拿到 id 这个名称的原因。iBd28资讯网——每日最新资讯28at.com

现在的问题变成了 writeReplace 方法究竟是个什么方法?iBd28资讯网——每日最新资讯28at.com

2. writeReplace

这个方法其实是系统底层自动生成的。我们可以将 Lambda 表达式在运行时生成的字节码保存下来,然后进行反编译,这样就能够看到 writeReplace 方法了。iBd28资讯网——每日最新资讯28at.com

如果需要将 Lambda 运行时生成的字节码保存,需要在启动参数中添加如下内容:iBd28资讯网——每日最新资讯28at.com

-Djdk.internal.lambda.dumpProxyClasses=/Users/sang/workspace/code/mp_demo/lambda/

等于号后面的部分是指定生成的字节码的保存位置,大家可以根据自己的实际情况去配置。iBd28资讯网——每日最新资讯28at.com

以本文一开头的 Lambda 表达式为例,最终生成的字节码反编译之后,内容如下:iBd28资讯网——每日最新资讯28at.com

final class MpDemo02ApplicationTests$$Lambda$1164 implements SFunction {    private MpDemo02ApplicationTests$$Lambda$1164() {    }    public Object apply(Object var1) {        return ((Book)var1).getId();    }    private final Object writeReplace() {        return new SerializedLambda(MpDemo02ApplicationTests.class, "com/baomidou/mybatisplus/core/toolkit/support/SFunction", "apply", "(Ljava/lang/Object;)Ljava/lang/Object;", 5, "org/javaboy/mp_demo02/model/Book", "getId", "()Ljava/lang/Integer;", "(Lorg/javaboy/mp_demo02/model/Book;)Ljava/lang/Object;", new Object[0]);    }}

大家可以看到,apply 方法实际上是重写的接口的方法,在这个方法中将传入的对象强转为 Book 类型,然后调用其 getId 方法。iBd28资讯网——每日最新资讯28at.com

然后大家看到,反编译之后多了一个 writeReplace 方法,这个方法的返回值是一个 SerializedLambda,这个 SerializedLambda 对象其实就是对 Lambda 表达式的描述。基本上每个参数都能做到见名知意,我这里说一下第七个参数,值是 getId,这个参数的变量名是 implMethodName,这就是我们 Lambda 表达式中给出来的变量名。这也是第一小节中,meta.getImplMethodName() 所获取到的值。iBd28资讯网——每日最新资讯28at.com

这下就清楚了,为什么写了 Book::getId 就能拿到属性名了。iBd28资讯网——每日最新资讯28at.com

3. 扩展知识

有的小伙伴注意到,在 qw.eq(Book::getId, 2); 方法中,第一个参数是一个 SFunction 的实例,那就说我直接给一个 SFunction 的实例,不用 Lambda。大家注意,这种写法不对!iBd28资讯网——每日最新资讯28at.com

原因在于经过前面的源码分析之后,我们发现,MP 中根据 Book::getId 去获取属性名称,一个关键点是利用 Lambda 在执行的时候生成的字节码去获取,如果你都没有用 Lambda,那也就不会生成所谓的 Lambda 字节码,也就不存在 writeReplace 方法,按照前文所分析的源码,就无法获取到属性名称。iBd28资讯网——每日最新资讯28at.com

还有小伙伴说,既然是 Lambda,那么我不用方法引用行不行?我像下面这样写行不行?iBd28资讯网——每日最新资讯28at.com

LambdaQueryWrapper<Book> qw = new LambdaQueryWrapper<>();qw.eq(b -> b.getId(), 2);List<Book> list = bookMapper.selectList(qw);System.out.println("list = " + list);

这也是一个 Lambda,但是如果你这样写了,运行之后就会报错。为什么呢?我们来看下这个 Lambda 生成的字节码反编译之后是什么样的:iBd28资讯网——每日最新资讯28at.com

final class MpDemo02ApplicationTests$$Lambda$1164 implements SFunction {    private MpDemo02ApplicationTests$$Lambda$1164() {    }    public Object apply(Object var1) {        return MpDemo02ApplicationTests.lambda$test18$3fed5817$1((Book)var1);    }    private final Object writeReplace() {        return new SerializedLambda(MpDemo02ApplicationTests.class, "com/baomidou/mybatisplus/core/toolkit/support/SFunction", "apply", "(Ljava/lang/Object;)Ljava/lang/Object;", 6, "org/javaboy/mp_demo02/MpDemo02ApplicationTests", "lambda$test18$3fed5817$1", "(Lorg/javaboy/mp_demo02/model/Book;)Ljava/lang/Object;", "(Lorg/javaboy/mp_demo02/model/Book;)Ljava/lang/Object;", new Object[0]);    }}

首先大家注意到 apply 方法生成的就不一样,apply 里边调用了 MpDemo02ApplicationTests.lambda$test18$3fed5817$1 方法,传入了 Book 对象作为参数。这个方法内容相当于就是 return book.getId();。然后在 writeReplace 方法中,返回 SerializedLambda 对象的时候,implMethodName 的值就是 lambda$test18$3fed5817$1 了。回到本文一开始的源码分析中,你会发现这样的方法名就无法提取出来我们想要的属性名。所以这种写法也不对。iBd28资讯网——每日最新资讯28at.com

从这里大家也可以看到,类似于 b -> b.getId() 这样的 Lambda,和方法引用 Book::getId 在底层是不同的。iBd28资讯网——每日最新资讯28at.com

再给小伙伴们举个例子,比如下面一段代码:iBd28资讯网——每日最新资讯28at.com

public class Demo01 {    public static void main(String[] args) {        Consumer<String> out1 = System.out::println;        out1.accept("javaboy");        Consumer<String> out2 = s -> System.out.println(s);        out2.accept("江南一点雨");    }}

这里有两个输出,第一个是一个方法引用,第二个则是一个常规的 Lambda 表达式。这两个执行起来效果是一致的,但是底层原理不同。iBd28资讯网——每日最新资讯28at.com

先来看第一个底层生成的 Lambda 字节码:iBd28资讯网——每日最新资讯28at.com

final class Demo01$$Lambda$14 implements Consumer {    private final PrintStream arg$1;    private Demo01$$Lambda$14(PrintStream var1) {        this.arg$1 = var1;    }    public void accept(Object var1) {        this.arg$1.println((String)var1);    }}

可以看到,这里把 System.out 的值 PrintStream 作为构造函数的参数传进来赋值给 arg变量,当调用方法的时候,再调用1.println 方法将字符串输出。iBd28资讯网——每日最新资讯28at.com

对于第二个底层生成的 Lambda 字节码如下:iBd28资讯网——每日最新资讯28at.com

final class Demo01$$Lambda$16 implements Consumer {    private Demo01$$Lambda$16() {    }    public void accept(Object var1) {        Demo01.lambda$main$0((String)var1);    }}

可以看到,这里有一个新的 lambda$main$0 方法,这个方法的底层逻辑其实就是我们自定义 Lambda 的时候写的 System.out.println(s)。iBd28资讯网——每日最新资讯28at.com

3. 小结

好啦,一篇小文,和小伙伴们探讨下 MP 中 qw.eq(Book::getId, 2); 方法的底层逻辑。iBd28资讯网——每日最新资讯28at.com

本文链接:http://www.28at.com/showinfo-26-82751-0.html通过方法引用获取属性名的底层逻辑是什么?

声明:本网页内容旨在传播知识,若有侵权等问题请及时与本网联系,我们将在第一时间删除处理。邮件:2376512515@qq.com

上一篇: 深入理解C/C++指针的算术运算

下一篇: 一文读懂Cache一致性原理

标签:
  • 热门焦点
  • K60至尊版狂暴引擎2.0加持:超177万跑分斩获性能第一

    Redmi的后性能时代战略发布会今天下午如期举办,在本次发布会上,Redmi公布了多项关于和联发科的深度合作,以及新机K60 Ultra在软件和硬件方面的特性,例如:“K60 至尊版,双芯旗舰
  • 2023年Q2用户偏好榜:12+256G版本成新主流

    3月份的性能榜、性价比榜和好评榜之后,就要轮到2023年的第二季度偏好榜了,上半年的新机潮已经过去,最明显的肯定就是大内存和存储的机型了,另外部分中端机也取消了屏幕塑料支架
  • 0糖0卡0脂 旭日森林仙草乌龙茶优惠:15瓶到手29元

    旭日森林无糖仙草乌龙茶510ml*15瓶平时要卖为79.9元,今日下单领取50元优惠券,到手价为29.9元。产品规格:0糖0卡0脂,添加草本仙草汁,清凉爽口,富含茶多酚,保留
  • 十个可以手动编写的 JavaScript 数组 API

    JavaScript 中有很多API,使用得当,会很方便,省力不少。 你知道它的原理吗? 今天这篇文章,我们将对它们进行一次小总结。现在开始吧。1.forEach()forEach()用于遍历数组接收一参
  • Automa-通过连接块来自动化你的浏览器

    1、前言通过浏览器插件可实现自动化脚本的录制与编写,具有代表性的工具就是:Selenium IDE、Katalon Recorder,对于简单的业务来说可快速实现自动化的上手工作。Selenium IDEKat
  • 三言两语说透设计模式的艺术-单例模式

    写在前面单例模式是一种常用的软件设计模式,它所创建的对象只有一个实例,且该实例易于被外界访问。单例对象由于只有一个实例,所以它可以方便地被系统中的其他对象共享,从而减少
  • 使用AIGC工具提升安全工作效率

    在日常工作中,安全人员可能会涉及各种各样的安全任务,包括但不限于:开发某些安全工具的插件,满足自己特定的安全需求;自定义github搜索工具,快速查找所需的安全资料、漏洞poc、exp
  • 三星获批量产iPhone 15全系屏幕:苹果史上最惊艳直屏

    按照惯例,苹果将继续在今年9月举办一年一度的秋季新品发布会,有传言称发布会将于9月12日举行,届时全新的iPhone 15系列将正式与大家见面,不出意外的话
  • 3699元!iQOO Neo8 Pro顶配版今日首销:1TB UFS 4.0同价位唯一

    5月23日,iQOO推出了全新的iQOO Neo8系列,包含iQOO Neo8和iQOO Neo8 Pro两个版本,其中标准版搭载高通骁龙8+,而Pro版更是首发搭载了联发科天玑9200+旗舰
Top