作为一个从 PHP 转 Java 的人,发现 alibaba 的 arthas 很好用。通过 arthas 的 redefine 命令,可以像 PHP 一样,不用重新发布,就可以改变程序行为。
但是用多了,发现很多时候,我们就改了几行代码,甚至有的时候就添加了一行日志,就无法 redefine 了。提示:
redefine error! java.lang.UnsupportedOperationException: class redefinition failed: attempted to add a method
它提示我们新增加方法,那我们就看看是不是新增加了方法。通过 javap 来查看定义的方法:
这是老的类:
这是新的类:
对比之后发现,新的类,即本地编译的类,其中的 lambda 对应的方法名都是 lambda$getAllCity$0 这样的,最后的编号是从 0 开始的。
而旧的类,即现在在运行的类,其中的同一个 lambda 的方法名是 lambda$getAllCity$121,最后的编号是一个非常大的数字。
在仔细对比下,发现是 jdk 的版本问题,不同的 jdk 版本对与 lamdba 的处理可能不一致。
具体来说,线上编译的 jdk 版本是 1.8.0_66-b17, 而本地是 1.8.0_222-b10,而这两个版本对 lambda 对应的方法命名是不一样的。
首先,为了调试方便,写一个最小复现用例来看看:
// Compile.java // 编译LamdbaTest1.java和LamdbaTest2.java import javax.tools.*; import java.io.File; public class Compile { public static void main(String[] args) { String path1 = "/path/to/LamdbaTest1.java"; String path2 = "/path/to/LamdbaTest2.java"; JavaCompiler javaCompiler = ToolProvider.getSystemJavaCompiler(); DiagnosticCollector diagnostics = new DiagnosticCollector(); StandardJavaFileManager fileManager = javaCompiler.getStandardFileManager(diagnostics, null, null); Iterable<"text-align: center">private static void lambda$main$0();而换版本 1.8.0_66-b17(旧版本 jdk)之后,lambda 的方法就成了:
private static void lambda$main$1();多尝试几个文件同时编译,我们就可以发现:对于旧版本的 javac,末尾这个数字是全局递增的,50 个类有 100 个 lambda,那最后一个 lambda 的编号就是 99;而新的版本是每个类重新计数的,和总共多少个类没有关系。
确认了问题之后,接下来就是不断的打断点、重试了。后来发现不同版本的 javac 逻辑确实不同。
首先,查看 jdk 源码可以知道,lambda 的方法名都是:
lambda$<methodname>$<lambdaCount>代码见
LambdaToMethod.java
不同的地方在于: 新版本的 javac,在处理一个新的类的时候,会保存上一个 lambdaCount,后续再恢复:
而旧版本则没有这个逻辑:
这就说明旧版本的编译器确实是 lambda 全局编号的。
那,问题来了,这个行为是从哪个版本变掉的呢?
对比之后发现这个变更是 jdk8u74-b02 引入的。对应的 bug 是 https://bugs.openjdk.java.net/browse/JDK-8067422,基本上就是每个类内的 lambda 单独编号,确保编译顺序不会影响 lambda 的方法名字。
所以,解决方案很简单,升级编译环境的 jdk 版本就好。
非常巧合的是,前两天为了更好的适配 Docker 运行环境(通俗的讲,就是在容器内获取到 docker 的 cpu 配额,而不是物理机器的 cpu 数量),我找运维添加了一个新的j dk 版本 1.8.0_231-b11,这样只需要直接将编译环境的 jdk 版本切换到 8u231 就行!
推荐使用 Arthas
"external nofollow" href="http://t.tb.cn/2A5CbHWveOXzI7sFakaCw8">http://t.tb.cn/2A5CbHWveOXzI7sFakaCw8
"external nofollow" href="https://github.com/alibaba/arthas">https://github.com/alibaba/arthas。
Arthas 第二期征文活动火热进行中,欢迎参加征文活动,还有奖品拿哦~点击了解详情。
总结
免责声明:本站资源来自互联网收集,仅供用于学习和交流,请遵循相关法律法规,本站一切资源不代表本站立场,如有侵权、后门、不妥请联系本站删除!
P70系列延期,华为新旗舰将在下月发布
3月20日消息,近期博主@数码闲聊站 透露,原定三月份发布的华为新旗舰P70系列延期发布,预计4月份上市。
而博主@定焦数码 爆料,华为的P70系列在定位上已经超过了Mate60,成为了重要的旗舰系列之一。它肩负着重返影像领域顶尖的使命。那么这次P70会带来哪些令人惊艳的创新呢?
根据目前爆料的消息来看,华为P70系列将推出三个版本,其中P70和P70 Pro采用了三角形的摄像头模组设计,而P70 Art则采用了与上一代P60 Art相似的不规则形状设计。这样的外观是否好看见仁见智,但辨识度绝对拉满。
更新日志
- dnf110版本刃影装备搭配
- 短剧《跟着微短剧去旅行》开机仪式在贵阳举行
- 你要是哭的话我就不听你说了,本来就和我没关系。
- 【游新视界】《印蒂卡》试玩 修女与撒旦的送信之旅
- [ABC]纯音乐发烧试音碟《开盘母带试音极品(6N纯银版)》[WAV+CUE]
- [ABC]圣路易诹坊太古社《响鼓》[6N纯银镀膜][正版低速原抓WAV+CUE]
- [ABC]《当·威廉姆斯发烧精选[6N纯银镀膜]》[正版低速原抓WAV+CUE]
- 空包叫花鸡店主称临时工错拿样品:目前已停售
- 成绩出色!托比版《蜘蛛侠》三部曲重映票房320万美元
- 风水轮流转!《地狱潜者2》单日获近3万好评
- 新手入坑指南推荐,新手战力养成指南介绍
- 原画壁纸及美图第211期,无水印可自取
- 原画壁纸及美图第211期,无水印可自取
- 王中平.2006-平凡的幸福【麻辣唱片】【WAV+CUE】
- 金城武.1996-精选集Memories【EMI百代】【WAV+CUE】