lambda 表达式导致 Arthas 无法 redefine 的问题
这篇文章主要介绍了 lambda 表达式导致 Arthas 无法 redefine 的问题,本文通过图文实例相结合给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下。
通过 arthas 的 redefine 命令,可以做到不用重新发布,就可以改变程序行为。
但是用多了,发现很多时候,我们就改了几行代码,甚至有的时候就添加了一行日志,就无法 redefine 了。提示:
redefine error! java.lang.UnsupportedOperationException: class redefinition failed: attempted to add a method
它提示我们新增加方法,那我们就看看是不是新增加了方法。通过 javap 来查看定义的方法:
这是老的类:
这是新的类:
对比之后发现,新的类,即本地编译的类,其中的 lambda 对应的方法名都是 lambda
0 这样的,最后的编号是从 0 开始的。
而旧的类,即现在在运行的类,其中的同一个 lambda 的方法名是 lambda
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<? extends JavaFileObject> compilationUnits = fileManager.getJavaFileObjects(
new File(path1),
new File(path2)
);
JavaCompiler.CompilationTask task = javaCompiler.getTask(null, fileManager, diagnostics, null, null,
compilationUnits);
boolean success = task.call();
System.out.println(success);
}
}
//LamdbaTest1.java
public class LamdbaTest1 {
private void test(Runnable runnable) {
runnable.run();
}
private void main() throws Throwable {
test(() -> {
System.out.println(11);
});
}
}
//LamdbaTest2.java
public class LamdbaTest2 {
private void test(Runnable runnable) {
runnable.run();
}
private void main() throws Throwable {
test(() -> {
System.out.println(22);
});
}
}
使用 1.8.0_222-b10(新版本 jdk)跑完了之后,发现 LamdbaTest2 中的 lambda 方法是:
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 就行!
原文出处:https://m.jb51.net/article/188155.htm 作者:鲁严波
- END -
- SpringBoot解决ajax跨域问题
- WebBrowser(IE) 与 JS 相互调用
- HOSTS配置问题导致集群异常故障分析
- Linux Regulator Framework(2)_regulator driver
- systemd的作用
- alsa声卡分析alsa-utils调用过程(二)-tinymixer
- alsa声卡分析alsa-utils调用过程(一)-tinyplay
- ALSA声卡驱动的DAPM(二)-建立过程
- ALSA声卡驱动的DAPM(一)-DPAM详解
- 高通Audio中ASOC的codec驱动(二)
- vue项目里的日期格式化
- CentOS下的Mysql的安装和使用
- vue路由跳转传参数
- 刘寅:TiDB 工具链和生态
- JavaScript 教程
- JavaScript 编辑工具
- JavaScript 与HTML
- JavaScript 与Java
- JavaScript 数据结构
- JavaScript 基本数据类型
- JavaScript 特殊数据类型
- JavaScript 运算符
- JavaScript typeof 运算符
- JavaScript 表达式
- JavaScript 类型转换
- JavaScript 基本语法
- JavaScript 注释
- Javascript 基本处理流程
- Javascript 选择结构
- Javascript if 语句
- Javascript if 语句的嵌套
- Javascript switch 语句
- Javascript 循环结构
- Javascript 循环结构实例
- Javascript 跳转语句
- Javascript 控制语句总结
- Javascript 函数介绍
- Javascript 函数的定义
- Javascript 函数调用
- Javascript 几种特殊的函数
- JavaScript 内置函数简介
- Javascript eval() 函数
- Javascript isFinite() 函数
- Javascript isNaN() 函数
- parseInt() 与 parseFloat()
- escape() 与 unescape()
- Javascript 字符串介绍
- Javascript length属性
- javascript 字符串函数
- Javascript 日期对象简介
- Javascript 日期对象用途
- Date 对象属性和方法
- Javascript 数组是什么
- Javascript 创建数组
- Javascript 数组赋值与取值
- Javascript 数组属性和方法
- python第十四课--排序及自定义函数之自定义函数(案例四)
- python第十四课--排序及自定义函数之自定义函数(案例五)
- python第十五课——全局变量and局部变量
- python第十六课——ascii码
- python第十六课——外部函数and内部函数
- python第十七课——列表生成式
- python第十八课——常用内置函数
- python第十九课——random模块中的常用函数
- python第二十课——math模块中常用的函数
- python第二十一课——str中的常用函数(重要)
- python第二十二课——list函数
- python第二十三课——dict中的函数
- python第二十四课——set中的函数
- python第二十五课——闭包
- LNMP架构应用实战——PHP扩展及缓存插件安装