001. 为啥用IDEA反编译没有擦除泛型?

news/2024/7/16 7:10:21 标签: intellij-idea, java, ide, 泛型, 泛型擦除

在这里插入图片描述

你好,我是YourBatman:一个俗人,贪财好色。

📚前言

Java泛型是进阶高级开发必备技能之一,了解实现泛型的基本原理,有助于写出更优质的代码。

众所周知,Java是伪泛型,是通过类型擦除(Type Erasure)来实现的。为了“查看/证明”Java对泛型类型的擦除,我们常常通过反编译的手段实现。Intellij IDEA作为Java开发主流IDE,它内置的反编译功能是最为常用的反编译工具。

但是,你会发现,IDEA的反编译竟没有擦除泛型

✍正文

如下代码:

java">/**
 * 在此处添加备注信息
 *
 * @author YourBatman's home page. <a href=https://yourbatman.cn>https://yourbatman.cn</a>
 * @author YourBatman. <a href=mailto:yourbatman@aliyun.com>Send email to me</a>
 * @author wechat:fsx641385712
 * @since 0.0.1
 */
public class Tester {

    @Test
    public void fun() {
        List<Integer> numbers = new ArrayList<>();
        numbers.add(18);

        List newNumbers = numbers;
        newNumbers.add("YourBatman");

        System.out.println(numbers);
    }

    @Test
    public void fun1() {
        List<Integer> intList = new ArrayList<>();
        List<String> stringList = new ArrayList<>();

        System.out.println(intList.getClass() == stringList.getClass());
    }

}

我们借助IDEA的反编译后的内容:找到需要反编译的.class文件
在这里插入图片描述
双击即可查看:
在这里插入图片描述
我的天,泛型类型不应该被擦除了吗,为毛还在?IDEA的反编译工具难道有bug?

🌈尝试其它反编译工具

IDEA最初内置的是著名的JD-GUI反编译插件,从2016年起改为自研的反编译插件Java Bytecode Decompiler,一直沿用至今:
在这里插入图片描述
为了验证此问题,我计划多试试几款反编译工具。

🚀jd-gui

下载地址:https://github.com/java-decompiler/jd-gui/releases
在这里插入图片描述
尴尬的是,双击打不开:在这里插入图片描述
无奈。在虚拟机里启了个Windows 11来跑:
在这里插入图片描述
结论:没有擦除泛型类型。和IDEA不同的是它反编译出来的结果更“原始”一丢丢

🚀jadx

下载地址:https://github.com/skylot/jadx/releases
在这里插入图片描述
同样的Windows 11上运行进行反编译:
在这里插入图片描述
结论:没有擦除泛型类型。结果不说和IDEA差不多,也是一模一样。

🚀JAD

下载地址:https://varaneckas.com/jad
在这里插入图片描述
由于我的本是基于Apple Silicon芯片的,所以只能继续在Windows上执行了:
在这里插入图片描述
结论:泛型类型被擦除了

🚀Beyond Compare 4

Beyond Compare的主业是做文件比较,其实它也可以Java反编译。只需在https://www.scootersoftware.com/download.php?zz=moreformats下载所需插件:
在这里插入图片描述
使用Beyond Compare 4进行反编译:
在这里插入图片描述
结论:泛型类型被擦除了。Beyond Compare 4的反编译基于Jad,因此效果和Jad一模一样

javap_c_79">🚀javap -c

使用最底层的javap -c进行反编译:
在这里插入图片描述
结论:泛型类型被擦除了

🍞总结

有些擦除了但有些没有擦除泛型类型,到底该信谁呢?当然是无条件相信javap -c,因为一切反编译操作都基于它。so结论是:Java的泛型是伪泛型,编译后泛型类型都会被擦除。

记住结论的同时,通过本文对比了多个反编译器的结果亦可得到两条基本的常识:

  1. 像IDEA内置的Java Bytecode Decompiler以及jadx这种比较新(还在持续迭代)的工具,称作智能反编译器更为合适:它能重排序代码,并且“保留”住泛型类型,方便开发者阅读
  2. Java泛型引入至今已有近20年,“伪泛型”已被认为是所有开发者的共识,没有必要再在反编译后体现出来反倒大大降低了可读性。像Jad这种“上古”时期的反编译器,依旧原汁原味

推荐阅读

  • IntelliJ IDEA 2022.3正式发布,配置云同步&支持Redis好用到炸

在这里插入图片描述
本专栏源代码库:https://github.com/yourbatman/yourbatman-999-question

  • 个人博客:https://yourbatman.cn
  • 程序员网盘:https://wangpan.yourbatman.cn
  • 女娲工程:https://start.yourbatman.cn
  • 更多专栏:https://yourbatman.cn/columns |或| 公号后台回复“专栏列表”获取全部小而美的原创技术专栏

我是YourBatman,一个俗人,贪财好色。历经过延期毕业、卖保险、送外卖的大龄程序员,《梦幻西游》骨灰玩家;龙珠迷、火影迷。前大厂资深技术专家,现资深领域建模专家、Java架构师;高质量代码、DDD面向对象设计布道师;Spring开源贡献者,CSDN博客之星年度Top 10,出版书籍《Spring奇淫巧技》&《领域建模之面向对象程序设计》进行时。wx:yourbatman-u


http://www.niftyadmin.cn/n/347706.html

相关文章

Android和iOS双端赞奇超级云盘APP公测版正式上线!

赞奇云工作站自发布以来&#xff0c;经过层层迭代和升级&#xff0c;以云工作站、赞奇超级云盘、软件中心、云渲染等功能&#xff0c;更高效地整合打通各行设计业务全流程&#xff0c;实现云上数字内容创作的完美呈现&#xff0c;取得了广大用户的喜爱和认可。 现在&#xff0…

软件测试被00后整顿职场了?

00后带来的压力 公司一位工作3年的老油条工资还没有刚来的00后高&#xff0c;她心中不平&#xff0c;对这件事情有不小的怨气&#xff0c;她觉得自己来公司三年了&#xff0c;三年内迟到次数都不超过5次&#xff0c;每天勤勤恳恳&#xff0c;要加班的时候也愿意加班&#xff0…

数据分析与预处理常用的图和代码

1.训练集和测试集统计数据描述之间的差异作图&#xff1a; def diff_color(x):color red if x<0 else (green if x > 0 else black)return fcolor: {color}(train.describe() - test.describe())[features].T.iloc[:,1:].style\.bar(subset[mean, std], alignmid, colo…

Cam APP-HAL流程追踪之demo梳理

一、基础知识 1、Google官网的Cam流程如下图1 2、Cam的预览、拍照、录像是分开的 Cam的预览、拍照、录像是各自独立的-换句话说可以不开启预览拍照或者录像–后面代码会详细介绍&#xff1b;市场上的成品Cam应用&#xff0c;打开Cam后直接打开了预览&#xff0c;然后可以拍照…

测试跳槽一次涨4k,我5年跳了3次...

最近有人说&#xff0c;现在测试岗位初始工资太低了&#xff0c;有些刚刚入行的程序员朋友说自己工资连5位数都没有.....干了好几年也没怎么涨。看看别人动辄月薪1.5到2万&#xff0c;其实我想说也没那么难。 说下如何高效地拿到2万的工资&#xff0c;总体来说&#xff0c;就靠…

2023ACP世界大赛中国总决赛|让世界再多一个微笑

5月21日&#xff0c;正值第三十三次全国助残日&#xff0c;作为公益推行的一份子&#xff0c;恒利联创也呈现出了“仁者爱人”的文化内核。 恒利联创携手微笑明天慈善基金会合作同行&#xff0c;旨在推动公益&#xff0c;促进残疾人事业的全面发展。在前行的道路上&#xff0c…

MRR 优化[Multi-Range Read optimization] (MySQL5.6)

BKA&#xff1a;SQL通过辅助索引要访问表数据时候&#xff0c;将大量的随机访问放入缓存&#xff0c;交给MRR接口合并为顺序访问。 MRR&#xff1a;在BKA算法应用之后&#xff0c;通过MRR接口合并随机访问为顺序访问&#xff0c;再去检索表数据。变大量随机为顺序访问。在通过辅…

一文了解customRef 自定义ref使用

概念 按照文档中的说明&#xff1a;customRef 可以用来创建一个自定义的 ref&#xff0c;并对其依赖项跟踪和更新触发进行显式控制。它需要一个工厂函数&#xff0c;该函数接收 track 和trigger函数作为参数&#xff0c;并且应该返回一个带有 get 和 set 的对象。 其实大致意思…