网易首页 > 网易号 > 正文 申请入驻

【238期】Java 8 中 Lambda 实现原理及源码剖析!

0
分享至

点击上方“Java精选”,选择“设为星标”

别问别人为什么,多问自己凭什么!

下方有惊喜,留言必回,有问必答!

每天08:15更新文章,每天进步一点点...

为了支持函数式编程,Java 8引入了Lambda表达式,那么在Java 8中到底是如何实现Lambda表达式的呢? Lambda表达式经过编译之后,到底会生成什么东西呢? 在没有深入分析前,让我们先想一想,Java 8中每一个Lambda表达式必须有一个函数式接口与之对应,函数式接口与普通接口的区别,可以参考前面的内容,那么你或许在想Lambda表达式是不是转化成与之对应的函数式接口的一个实现类呢,然后通过多态的方式调用子类的实现呢,如下面代码是一个Lambda表达式的样例:


@FunctionalInterface
interface Print {
public void print(T x);
public class Lambda {
public static void PrintString(String s, Print print) {
print.print(s);
public static void main(String[] args) {
PrintString("test", (x) -> System.out.println(x));


按照上面的分析,理论上经过编译器处理后,最终生成的代码应该如下面所示:


@FunctionalInterface
interface Print {
public void print(T x);

class Lambda$$0 implements Print {
@Override
public void print(String x) {
System.out.println(x);
}
}

public class Lambda {
public static void PrintString(String s,
Print print) {
print.print(s);
}
public static void main(String[] args) {
PrintString("test", new Lambda$$0());
}
}

再或者是一个内部类实现,代码如下所示:


@FunctionalInterface
interface Print {
public void print(T x);
public class Lambda {
final class Lambda$$0 implements Print {
@Override
public void print(String x) {
System.out.println(x);

public static void PrintString(String s,
Print print) {
print.print(s);
public static void main(String[] args) {
PrintString("test", new Lambda().new Lambda$$0());


异或是这种匿名内部类实现,代码如下所示:


@FunctionalInterface
interface Print {
public void print(T x);
public class Lambda {
public static void PrintString(String s,
Print print) {
print.print(s);
public static void main(String[] args) {
PrintString("test", new Print() {
@Override
public void print(String x) {
System.out.println(x);



上面的代码,除了在代码长度上长了点外,与用 Lambda 表达式实现的代码运行结果是一样的,那么 Java 8 到底是用什么方式实现的呢? 是不是上面三种实现方式中的一种呢,你也许觉的自已想的是对的,其实本来也就是对的,在 Java 8 中采用的是内部类来实现 Lambda 表达式。

为了探究 Lambda 表达式是如何实现的,就得需要研究 Lambda 表过式最终转化成的字节码文件,这就需要 jdk 的 bin 目录下的一个字节码查看工具及反编译工具:

javap -p Lambda.class

上面命令中的 -p 表示输出所有类及成员,运行上面的命令后,得的结果如下所示:


Compiled from "Lambda.java"
public class Lambda {
public Lambda();
public static void PrintString(java.lang.String, Print);
public static void main(java.lang.String[]);
private static void lambda$0(java.lang.String);

由上面的代码可以看出编译器会根据 Lambda 表达式生成一个私有的静态函数,注意,在这里说的是生成,而不是等价:

private static void lambda$0(java.lang.String);

为了验证上面的转化是否正确? 我们在代码中定义一个 lambda$0 这个的函数,最终代码如下所示:


@FunctionalInterface
interface Print {
public void print(T x);

public class Lambda {
public static void PrintString(String s,
Print print) {
print.print(s);
}
private static void lambda$0(String s) {
}
public static void main(String[] args) {
PrintString("test", (x) -> System.out.println(x));
}
}

上面的代码在编译时不会报错,但是运行时就会报错,因为存在两个 lambda$0 函数,如下所示,是运行时的错误:


Exception in thread "main" java.lang.ClassFormatError: Duplicate method name&signature in class file Lambda
at java.lang.ClassLoader.defineClass1(Native Method)
at java.lang.ClassLoader.defineClass(ClassLoader.java:760)
at java.security.SecureClassLoader.defineClass(SecureClassLoader.java:142)
at java.net.URLClassLoader.defineClass(URLClassLoader.java:467)
at java.net.URLClassLoader.access$100(URLClassLoader.java:73)
at java.net.URLClassLoader$1.run(URLClassLoader.java:368)
at java.net.URLClassLoader$1.run(URLClassLoader.java:362)
at java.security.AccessController.doPrivileged(Native Method)
at java.net.URLClassLoader.findClass(URLClassLoader.java:361)
at java.lang.ClassLoader.loadClass(ClassLoader.java:424)
at sun.misc.Launcher$AppClassLoader.loadClass(Launcher.java:331)
at java.lang.ClassLoader.loadClass(ClassLoader.java:357)
at sun.launcher.LauncherHelper.checkAndLoadMain(LauncherHelper.java:495)

通过 javap 对上述错误代码进行反编译,反编译之后输出的类的成员如下所示:


Compiled from "Lambda.java"
public class Lambda {
public Lambda();
public static void PrintString(java.lang.String, Print);
private static void lambda$0(java.lang.String);
public static void main(java.lang.String[]);
private static void lambda$0(java.lang.String);

会发现 lambda$0 出现了两次,那么在代码运行的时候,就不知道去调用哪个,因此就会抛错。

有了上面的内容,可以知道的是 Lambda 表达式在 Java 8 中首先会生成一个私有的静态函数,这个私有的静态函数干的就是 Lambda 表达式里面的内容,因此上面的代码初步可以转化成如下所示的代码:


@FunctionalInterface
interface Print {
public void print(T x);
public class Lambda {
public static void PrintString(String s, Print print) {
print.print(s);

private static void lambda$0(String x) {
System.out.println(x);
}

public static void main(String[] args) {
PrintString("test", /**lambda expression**/);
}
}

转化成上面的形式之后,那么如何实现调用静态的 lambda$0 函数呢,在这里可以在以下方法打上断点,可以发现在有 lambda 表达式的地方,运行时会进入这个函数:


public static CallSite metafactory(MethodHandles.Lookup caller,
String invokedName,
MethodType invokedType,
MethodType samMethodType,
MethodHandle implMethod,
MethodType instantiatedMethodType)
throws LambdaConversionException {
AbstractValidatingLambdaMetafactory mf;
mf = new InnerClassLambdaMetafactory(caller, invokedType,
invokedName, samMethodType,
implMethod, instantiatedMethodType,
false, EMPTY_CLASS_ARRAY, EMPTY_MT_ARRAY);
mf.validateMetafactoryArgs();
return mf.buildCallSite();

在这个函数中可以发现为 Lambda 表达式生成了一个内部类,为了验证是否生成内部类,可以在运行时加上 -Djdk.internal.lambda.dumpProxyClasses,加上这个参数后,运行时,会将生成的内部类 class 码输出到一个文件中:


final class Lambda$$Lambda$1 implements Print {
private Lambda$$Lambda$1();
public void print(java.lang.Object);

如果运行 javap -c -p 则结果如下:


final class Lambda$$Lambda$1 implements Print {
private Lambda$$Lambda$1();
Code:
0: aload_0
1: invokespecial #10 // Method java/lang/Object."":()V
4: return

public void print(java.lang.Object);
Code:
0: aload_1
1: checkcast #14 // class java/lang/String
4: invokestatic #20 // Method Lambda.lambda$0:(Ljava/lang/String;)V
7: return
}

通过上面的字节码指令可以发现实现上调用的是 Lambda.lambda$0 这个私有的静态方法。

因此最终的 Lambda 表达式等价于以下形式:


@FunctionalInterface
interface Print {
public void print(T x);
public class Lambda {
public static void PrintString(String s, Print print) {
print.print(s);
private static void lambda$0(String x) {
System.out.println(x);
final class $Lambda$1 implements Print{
@Override
public void print(Object x) {
lambda$0((String)x);

public static void main(String[] args) {
PrintString("test", new Lambda().new $Lambda$1());


如上就是实现原理了!

作者:让猪再飞会 https://www.cnblogs.com/WJ5888/p/4667086.html

公众号“Java精选”所发表内容注明来源的,版权归原出处所有(无法查证版权的或者未注明出处的均来自网络,系转载,转载的目的在于传递更多信息,版权属于原作者。如有侵权,请联系,笔者会第一时间删除处理!

------ THE END ------

精品资料,超赞福利!


3000+ 道面试题在线刷,最新、最全 Java 面试题!

期往精选 点击标题可跳转

技术交流群!

最近有很多人问,有没有读者&异性交流群,你懂的!想知道如何加入。加入方式很简单,有兴趣的同学,只需要点击下方卡片,回复“加群”,即可免费加入交流群!

文章有帮助的话,在看,转发吧!

特别声明:以上内容(如有图片或视频亦包括在内)为自媒体平台“网易号”用户上传并发布,本平台仅提供信息存储服务。

Notice: The content above (including the pictures and videos if any) is uploaded and posted by a user of NetEase Hao, which is a social media platform and only provides information storage services.

相关推荐
热点推荐
惊魂!深圳一小汽车突发意外,失控冲出道路!车体4/5悬空,在悬崖边缘摇摇欲坠…

惊魂!深圳一小汽车突发意外,失控冲出道路!车体4/5悬空,在悬崖边缘摇摇欲坠…

广东活动
2025-12-27 12:11:49
揭穿南京博物院背后的徐小姐“画皮”,这就是打蛇能打的“七寸”

揭穿南京博物院背后的徐小姐“画皮”,这就是打蛇能打的“七寸”

人格志
2025-12-27 23:44:32
人民战争!高市支持率还有67%,中国100%取消未来两周中日航班

人民战争!高市支持率还有67%,中国100%取消未来两周中日航班

离离言几许
2025-12-28 07:01:35
20岁泰国王子提帮功拜见僧王,40岁贵妃诗妮娜相伴,却也“卑微”

20岁泰国王子提帮功拜见僧王,40岁贵妃诗妮娜相伴,却也“卑微”

译言
2025-12-28 06:38:21
目前见过的最清纯春丽cos,赛场遇见也舍不得用力

目前见过的最清纯春丽cos,赛场遇见也舍不得用力

街机时代
2025-12-27 15:00:03
卖爆了!单月出口13.2万辆,反超上汽集团,是特斯拉的7倍

卖爆了!单月出口13.2万辆,反超上汽集团,是特斯拉的7倍

简易科技
2025-12-25 15:57:25
票房会破20亿!宁浩《用武之地》首轮口碑出炉,徐峥的话又应验了

票房会破20亿!宁浩《用武之地》首轮口碑出炉,徐峥的话又应验了

小丸子的娱乐圈
2025-12-26 17:59:11
海南封关运作后,就相当于中国版的新加坡正式上线了。

海南封关运作后,就相当于中国版的新加坡正式上线了。

流苏晚晴
2025-12-21 16:05:12
想要6年不换新,2025公认接近完美的3款手机,16GB+512GB重点推荐

想要6年不换新,2025公认接近完美的3款手机,16GB+512GB重点推荐

科技阿维
2025-12-26 18:37:49
“坚持计划生育一百年不动摇”的彭佩云去世,她父亲的人生更传奇

“坚持计划生育一百年不动摇”的彭佩云去世,她父亲的人生更传奇

文史微鉴
2025-12-24 23:47:43
直击:日本发生重大交通事故,67辆车连环撞,幸存者颤抖回忆惨烈现场……

直击:日本发生重大交通事故,67辆车连环撞,幸存者颤抖回忆惨烈现场……

日本物语
2025-12-27 20:32:22
趋之若鹜!上海海港成冬窗香饽饽:中超球员争相排队进港

趋之若鹜!上海海港成冬窗香饽饽:中超球员争相排队进港

足球大腕
2025-12-27 20:40:13
《老舅》大结局:最恶毒的不是张秘书、汤伟,是只出场一次的老刘

《老舅》大结局:最恶毒的不是张秘书、汤伟,是只出场一次的老刘

宇林网络
2025-12-27 17:35:23
冬至后30天是补阳“黄金期”,常吃这3样,寒痰湿或被慢慢清走

冬至后30天是补阳“黄金期”,常吃这3样,寒痰湿或被慢慢清走

江江食研社
2025-12-25 16:30:07
银行女经理自杀,年仅36岁,知情人透不堪内幕,众多网友炸锅了

银行女经理自杀,年仅36岁,知情人透不堪内幕,众多网友炸锅了

原梦叁生
2025-12-27 21:50:06
中国00后斯诺克“三剑客”常冰玉、吴宜泽和斯佳辉,未来震惊台坛

中国00后斯诺克“三剑客”常冰玉、吴宜泽和斯佳辉,未来震惊台坛

金风说
2025-12-28 06:38:41
中亚铁娘子:赶跑两任总统后自己当总统,最大愿望就是与中国合作

中亚铁娘子:赶跑两任总统后自己当总统,最大愿望就是与中国合作

梁蜱爱玩车
2025-12-26 22:20:44
重磅!亚裔非法移民或被驱逐到太平洋小岛去打鱼

重磅!亚裔非法移民或被驱逐到太平洋小岛去打鱼

大洛杉矶LA
2025-12-28 01:22:28
朱芳雨一家近照,42岁身家过亿,二婚妻子好美,儿女双全很幸福

朱芳雨一家近照,42岁身家过亿,二婚妻子好美,儿女双全很幸福

大西体育
2025-12-23 10:26:59
不演了!乔欣晒圣诞宅家聚会,终于干回富家千金老本行了

不演了!乔欣晒圣诞宅家聚会,终于干回富家千金老本行了

玲儿爱唱歌
2025-12-26 12:06:22
2025-12-28 08:15:00
Java精选
Java精选
一场永远也演不完的戏
1766文章数 3859关注度
往期回顾 全部

科技要闻

小米也涨价了!业界称终端再不涨明年必亏

头条要闻

山西大同一小区物业禁止新能源车进入地库:担心自燃

头条要闻

山西大同一小区物业禁止新能源车进入地库:担心自燃

体育要闻

83分钟绝杀!曼城2-1年度收官:英超6连胜狂飙

娱乐要闻

张昊唯逃税涉黄风波落幕:法院认定朋友造谣

财经要闻

注意,开始拉物价了!

汽车要闻

好音响比大屏更重要?车企开始“听”用户的

态度原创

本地
艺术
旅游
游戏
手机

本地新闻

云游安徽|踏访池州,读懂山水间的万年史书

艺术要闻

手串种类大盘点,全见过的算得上是文玩老手了!

旅游要闻

又添一条樱花大道 大理三月好风光

《歧路旅人0》评测:授予一切"/> 主站 商城 论坛 自运营 登录 注册 《歧路旅人0》评测:授予一切 星河 2025-12-27 返回专栏首页 作者:...

手机要闻

荣耀WIN系列全球首发10000mAh电池:友商短时间内很难超越

无障碍浏览 进入关怀版