作为Java码农一枚,不了解Java的Class字节码结构似乎情有可原。但是关于Java代码的编译优化,或许值得一探究竟。我们编写好一个Java类文件之后,通过javac命令将一个*.java文件编译为*.class字节码文件。同样我们可以将一个*.class文件反编译为一个*.java文件。通过研习反编译之后的Java代码和原始代码的不同,我们可以了解一些Java代码编译优化的基础知识,有助于我们编写出更加高效实用并且简洁易读的代码。
在这里,我们使用非常流行的DJ Java Decompiler作为反编译工具,DJ Java Decompiler操作简单、功能强大,操作界面如下所示,关于DJ Java Decompiler的使用方式,诸君参考官方文档即可。
【字符串拼接】
String s = "a" + "b" + "c"; System.out.println(s == "abc");
编写如上代码,编译运行之后返回结果为true,这让我们始料未及。通过反编译之后,我们发现
String s = "a" + "b" + "c";
在编译时进行了代码优化:
String s = "abc";
两个字符串常量进行比较,内容相同返回值为true,似乎不难理解。
【常量值计算】
int bufferSize = 2 * 1024 * 1024; System.out.println(bufferSize);
我们非常乐意编写如上所示的Java代码,定义一个变量bufferSize来表示缓冲区的大小,值为2m,一目了然。然后一些偏执狂似乎会觉得这样运行效率不高,不如直接定义
int bufferSize = 2097152; System.out.println(bufferSize);
通过对代码反编译之后,我们发现在编译期已经进行了常量值的运算、得出了结果,并且使用16进制进行表示、方便运行时进行存储,代码如下所示。这让我们感到的应该不仅仅是惊讶,更是对编译器设计者的肃然起敬。
int i = 0x200000; System.out.println(i);
【泛型擦除】
List<String> strList = new ArrayList<String>(); strList.add("a"); strList.add("bc"); strList.add("123");
通常我们使用泛型集合类,强制要求添加的集合中的元素是统一的类型。但是Java编译器在编译时进行了泛型擦除,运行时已经不存在任何关于集合类的泛型信息。因此在运行时我们可以往这个集合里面添加任何类型的Java对象,尽管这种方式不被推荐。反编译之后的代码如下所示:
ArrayList arraylist = new ArrayList(); arraylist.add("a"); arraylist.add("bc"); arraylist.add("123");
【for-each循环与StringBuilder】
String output = ""; for(String str : strList) { output += str; } System.out.println(output);
我们通常使用for-each循环对集合进行遍历,并且在循环中进行了字符串拼接操作。Java编译器对for-each循环进行了优化,使用迭代器Iterator来对集合进行遍历。
并且对于包含变量的字符串拼接操作,使用StringBuilder类的append()方法来完成。令人遗憾的是,在for-each循环中定义了局部的StringBuilder变量,并且使用toString()方法时又创建了String对象。在循环中创建大量对象是应该尽可能进行避免的,这些对象的生命周期很短,需要频繁的进行垃圾回收GC,加重了CPU的负担,降低了系统的运行效率。鉴于此,我们可以在for-each循环的外部定义StringBuilder变量,进行字符串的拼接操作。
String s1 = ""; for(Iterator iterator = arraylist.iterator(); iterator.hasNext();) { String s2 = (String)iterator.next(); s1 = (new StringBuilder()).append(s1).append(s2).toString(); } System.out.println(s1);
【装箱与拆箱】
Integer i = 2; i = i + 3; System.out.println(i == 5);
如上代码编译运行之后结果为true。我们很好奇,一个字面常量2究竟是如何赋值给一个Integer类型的引用变量的,并且Integer类型的引用变量i又是如何和常量3进行算术运算的?更让人不可思议的是,Integer引用变量表示的是对象的“内存地址”,又怎么会和5相等呢?将如上代码反编译之后,一切不再神秘:
Integer integer = Integer.valueOf(2); integer = Integer.valueOf(integer.intValue() + 3); System.out.println(integer.intValue() == 5);
=========================================
使用这种方式研习Java代码的编译优化,确实非常low,但是从编写代码的角度来考虑,确实成本也很low!
相关推荐
JavaDecompiler反编译是Java开发中的一种技术,主要用于将已编译的Java字节码(.class文件)转换回可读的源代码格式。这一过程对于开发者来说有着多种用途,比如逆向工程、学习库的内部实现或者调试已丢失源代码的类...
JavaDecompiler是一款强大的Java反编译工具,它允许开发者查看并理解Java字节码编译后的类文件。在软件开发过程中,有时我们需要查看已编译的.class文件的源代码,而原始的.java文件可能丢失或者不可用。这时,Java...
值得注意的是,尽管JavaDecompiler尽力还原源代码,但反编译结果可能并不完全与原始源代码一致。一些编译器优化可能会导致反编译后的代码难以理解,而且失去了原始的作者注释。此外,反编译行为应遵循版权法和软件...
在Java编程环境中,源代码被编译成字节码,这是一种中间表示,可以在Java虚拟机(JVM)上运行。然而,由于字节码并不直接对应于源代码,因此在某些情况下,我们可能需要反编译字节码来理解其工作原理或进行调试。...
- **其他功能**:支持代码优化、混淆代码分析、资源提取等高级功能。 3. **压缩包内容解析**: - **DJHNEW.chm**:这可能是一个帮助文件,通常以CHM(Microsoft Compiled HTML Help)格式存储,包含DJ Java ...
JavaDecompiler是一款强大的Java反编译工具,它能够将已编译的Java字节码(.class文件)转换回源代码格式,使开发者能够查看和理解已加密或未公开的Java程序的内部工作原理。这一过程对于软件逆向工程、学习第三方库...
2)反编译代码规范,无需优化,反编译不出来的代码直接输出错误信息,不像jd-gui经常会产生看似能用实际上却不能用的垃圾代码 3)对循环嵌套处理能力不错 缺点: 1)不像原生的jad反编译能力那么强大,有些类反...
在使用Java反编译器时,需要注意的是,由于编译过程中的优化,反编译后的代码可能与原始源代码有所不同,尤其是在方法体内的细节上。此外,如果原代码进行了混淆,反编译后的代码可能难以理解。 在实际应用中,Java...
DJ Java Decompiler就是这样一款工具,它可以将字节码反编译回接近原始的Java源代码格式,尽管反编译的代码可能无法完全与原始源代码一致,但通常足以理解大部分逻辑。 该软件的特性包括: 1. **用户友好的界面**...
5. **代码优化**:它能识别并处理Java字节码中的优化技术,比如内联常量和方法,以提供更清晰的反编译结果。 使用DJ JavaDecompiler,开发者可以: 1. **分析未知库**:当遇到没有源代码的第三方库时,可以使用此...
Java反编译器是开发者和逆向工程人员的利器,它允许我们将已编译的`.class`文件转换回源代码的`.java`格式。这对于理解已加密或混淆的代码、查看库函数的内部实现或者进行代码调试都非常有用。在本篇中,我们将深入...
- **代码优化**:通过查看反编译的代码,可能发现性能瓶颈并进行优化。 4. **使用注意事项:** - **版权问题**:反编译他人的代码可能存在版权风险,只有在合法且合理的情况下才能使用。 - **精度问题**:反编译...
Cavaj Java Decompiler是一款强大的JAVA反编译工具,它能够将Java字节码(.class文件)转换回源代码,使开发者可以查看并理解已编译的Java程序的内部结构。这一过程对于逆向工程、软件调试、学习库功能或者在没有源...
DJ Java Decompiler v3.7.7.81就是这样一款强大的Java反编译软件,它能帮助开发者从字节码还原出接近原始的Java源代码,便于理解和调试。 一、Java反编译的基本概念 Java反编译是将Java字节码(.class文件)转换回...
Cavaj Java Decompiler是一款专为Java开发者设计的小型反编译工具,它能够将已编译的Java字节码(.class文件)转换回源代码格式,使得开发者可以查看和理解原本的Java源代码。在软件开发、逆向工程、代码分析以及...
- 反编译并不总是能够完全恢复原始源代码,尤其是当原始代码进行了混淆或优化时。 - 使用反编译器应遵循版权法律,不得用于非法目的,尊重他人的知识产权。 总之,Procyon反编译器是Java和Groovy开发者的重要工具,...
尽管不能保证完全恢复原汁原味的源代码,尤其是当原始代码包含复杂优化或者非标准编程技巧时,但Java Decompiler 0.3.3已经做得相当出色,能够为大多数常见情况提供准确的反编译结果。 在实际使用中,开发者可以...
需要注意的是,由于Java字节码的优化和混淆,反编译的结果可能并非完全等同于原始的Java源代码,尤其是在处理包含复杂优化或闭包的代码时。此外,JD-GUI并不支持反编译Java的内省(introspection)和反射代码,也不...