`
zhaohaolin
  • 浏览: 1011379 次
  • 性别: Icon_minigender_1
  • 来自: 杭州
社区版块
存档分类
最新评论

Java字节码揭秘——第一部分

    博客分类:
  • JAVA
阅读更多
写在前面
这一两年,在 JVM 上使用其他替代语言越来越热门了。现在至少有三门语言有幸在 Java Community Process 中得到了官方认可: JRuby Groovy Bean-Shell 。另外,代号为野马 (Mustang) Java 6 发布了包含了一个专为封装不同脚本引擎的 API 层,就像 JDBC 访问数据库的模式一样。再加上 Java 版本 5 也在语言本身上做了很大的调整。总之,就像我之前翻译的一篇 BLOG 一样, Java 平台的编程语言的前景已经发生了巨大的改变。虽然如此,只有一样东西没有变,它是所有这些语言的基础,无论这些语言有多么吸引人的特性和功能,最终都会在 JVM 的混合语言中运行,即 JVM 字节码。这又提起了我在 JVM/Java 字节码方面的兴趣。所以书写本文,在其中将介绍 JVM 字节码集合,用一些代码来描述它的工作方式,也将介绍一些可以直接操纵字节码的工具。
 
首先我要说明的是,直接了解 JVM 字节码感觉是奇怪的事情,因为我们总不可能自己来书写字节码。但是,我们如果知道编译器干了些什么可能会更好一点。比如,你肯定想知道编译后的 StringBuffer String 的区别、编译器到底有没有给你加上默认构造函数……当你了解了 JVM 字节码——这是我看见过的最简单的“可装配语言”——你就能够验证你的这些假设是否正确。
 
 
分解 Java
考虑到大家对 Java 都已经比较熟悉了,所以我们这样开始可能比较容易:我们从编译后的 Java 代码开始,然后对其进行分解。这样可能比一开始就直接讲述 Java 字节码的规则要好一些。我们先从最简单的 Hello World 程序开始。
 
public class HelloWorld

{

        public static void main(String[] args)

        {

               System.out.println("Hello, world!");

        }

}

 
我们通过两种方式来一起研究 Java 字节码。第一个是太久时间都没有见到过的 javap javap 是字节码分解器,意思就是它编译 .class 文件并将文件结构输出到控制台,其中包括组成方法的字节码。如下例:
 
$ javap -verbose -c -private HelloWorld

Compiled from "HelloWorld.java"

public class HelloWorld extends java.lang.Object

        SourceFile: "HelloWorld.java"

        minor version: 0

        major version: 50

        Constant pool:

const #1 = Method #6.#15; // java/lang/Object."<init>":()V

const #2 = Field #16.#17; // java/lang/System.out:Ljava/io/PrintStream;

const #3 = String #18; // Hello, world!

const #4 = Method #19.#20; // java/io/PrintStream.println:(Ljava/lang/String;)V

const #5 = class #21; // HelloWorld

const #6 = class #22; // java/lang/Object

const #7 = Asciz <init>;

const #8 = Asciz ()V;

const #9 = Asciz Code;

const #10 = Asciz LineNumberTable;

const #11 = Asciz main;

const #12 = Asciz ([Ljava/lang/String;)V;

const #13 = Asciz SourceFile;

const #14 = Asciz HelloWorld.java;

const #15 = NameAndType #7:#8;// "<init>":()V

const #16 = class #23; // java/lang/System

const #17 = NameAndType #24:#25;// out:Ljava/io/PrintStream;

const #18 = Asciz Hello, world!;

const #19 = class #26; // java/io/PrintStream

const #20 = NameAndType #27:#28;// println:(Ljava/lang/String;)V

const #21 = Asciz HelloWorld;

const #22 = Asciz java/lang/Object;

const #23 = Asciz java/lang/System;

const #24 = Asciz out;

const #25 = Asciz Ljava/io/PrintStream;;

const #26 = Asciz java/io/PrintStream;

const #27 = Asciz println;

const #28 = Asciz (Ljava/lang/String;)V;

 

{

public HelloWorld();

        Code:

               Stack=1, Locals=1, Args_size=1

               0: aload_0

               1: invokespecial #1; //Method java/lang/Object."<init>":()V

               4: return

 

        LineNumberTable:

               line 1: 0

 

public static void main(java.lang.String[]);

        Code:

               Stack=2, Locals=1, Args_size=1

               0: getstatic #2; //Field java/lang/System.out:Ljava/io/PrintStream;

               3: ldc #3; //String Hello, world!

               5: invokevirtual #4; //Method java/io/PrintStream.println:(Ljava/lang/String;)V

               8: return

        LineNumberTable:

               line 5: 0

               line 6: 8

}

 
 
在刚才讲述的 .class 文件实际并不准确, JVM 无所谓输入的二进制流从哪儿来,只不过因为我们的习惯和 JDK 1.0 的发布所以我们说成是 .class 文件。所以,所谓的“ .class 文件”应该被理解为符合 JVM 标准的二进制格式流。
 
上面我们使用了 javap 。其中, -c 指示需要显示方法字节码; -private 指示无论可访问性显示所有成员; -verbose 是需要显示类的常量池。检查 HelloWorld 分解后的内容,会觉得非常有趣,我们立马就可以验证一些假设。例如,第一,如果类没有显式声明其父类的话,它将继承于 java.lang.Object 。第二, javap 也验证了如果类中没有显式声明构造函数的话,编译器会插入一个缺省无参的构造函数 ( 构造函数在 JVM 级别是显示成 <init> 的普通函数 )
 
加上了 -verbose 选项的 javap 输出中一个重要的部分就是常量池。每个类都会有个常量池,所有的常量——比如字符串、类名、方法名、属性名——都是保存在类的中心位置,通过对该池的索引进行参照访问。通常,这些特殊的细节内容都是由工具来处理的,这也是 javap 通过注释来显示这些常量值的原因。但是这些内容对我们认识常量池非常有用,也能够简化我们对分解代码的理解。例如,第 5 行代码 System.out.println("Hello, world!"); 它调用了 println 方法,显示在常量池的编号为 4 的分片 (const #4) ,它依次由编号为 19 的分片和编号为 20 的分片组成 (const #4 = Method #19.#20;) ,这样就最终解决了 java.io.PrintStream.println(String[]) 的问题。你可以 参照JVM标准 来了解所有不同的常量类型以及他们在 .class 文件中的格式。
 
在这里,我们主要来分析自动生成的 HelloWorld 构造函数:
 
public HelloWorld();

        Code:

               Stack=1, Locals=1, Args_size=1

               0: aload_0

               1: invokespecial #1; //Method java/lang/Object."<init>":()V

               4: return

        LineNumberTable:

               line 1: 0

 
JVM 中,所有字节码都是通过一个基本的原则来进行堆栈操作的:每个操作符可能会消费一个或多个操作计数,并可能最后将一个操作计数推送到执行堆栈。需要注意的是,每个分片 (slot) 都是 32 位的,这就意味着 long 或者是 double 的值会消耗两个分片 (slot)( 很多人认为这个是 JVM 实现中的最大缺憾 ) 。另外,每个方法都会有一个本地的结合,本地变量和参数都在此保存。因此,例如“ aload_0 ”指示符将第一个参数带入方法,并将其推送至执行堆栈。“ invokespecial ”指示符,不言而喻,它将调用实例的方法,但是忽略传统的动态绑定 ( 因为我们显示调用基类版本的覆盖方法,该特殊的操作符用在父“ super ”调用 ) 。因为 Object 的构造函数需要一个参数 (this 指针 ) ,所以它将消耗执行堆栈中的一个分片 ( 记住,这是我们刚才推送的参数—— this 指针,指向我们自己的实例的 this 指针 ) ,而且它不返回任何值 ( 最后有一个 V ) ,当方法返回时它将不往堆栈内推送任何内容。此时, HelloWorld 的构造函数已完成任务,所以它通过“ return ”操作符进行简单返回。
 
我们接下来在看看写在 HelloWorld 里面的主方法 (main)
 
public static void main(java.lang.String[]);

        Code:

               Stack=2, Locals=1, Args_size=1

               0: getstatic #2; //Field java/lang/System.out:Ljava/io/PrintStream;

               3: ldc #3; //String Hello, world!

               5: invokevirtual #4; //Method java/io/PrintStream.println:(Ljava/lang/String;)V

               8: return

        LineNumberTable:

               line 5: 0

               line 6: 8

 
因为它是静态方法,所以最显著的区别就是第一个参数并不是 this 指针,除此之外,它和 HelloWorld 的构造函数看起来都差不多。第一个操作符“ getstatic ”将获取一个 static 区域并将其值推送至堆栈中,在本例中是 System.out 的引用,由 #2 常量池分片描述,并在操作符后使用注释显示。接下来,就对字符串“ Hello, World! ”进行加载,它在 #3 常量池分片中存储。通过堆栈上的两个引用,我们就可以调用“ invokevirtual PrintStream.println(String[]) 方法了。因其需要一个参数,再加上调用该方法需要的初始 this 引用,我们刚才推送至堆栈的这两项就被消费了, println(String[]) 不返回任何值,所以完成后堆栈上就为空了。一个简单的“ return ”操作符中止了该方法,任务完成了。
 
后面的内容会比现在的复杂一些,但总的来说,了解 Java 字节码的重要部分是需要了解每个操作符是如何操作执行堆栈的。
未完待续……
参考资料下载:

The Java Virtual Machine Specification(2nd Edition) JVM规范(第二版)

分享到:
评论

相关推荐

    轻松看懂Java字节码.pdf

    1. 魔数(Magic Number):class文件开头的“cafebabe”是Java字节码文件的魔数,用于标识一个文件是有效的Java字节码文件。这一点在内容部分被明确指出。 2. 主版本号和次版本号:紧随魔数之后的是次版本号和主...

    JAVA字节码JAVA字节码.doc

    javap 是 JDK 中的一个命令行工具,用于将 Java 字节码反编译成可读的格式。javap 可以将类文件(.class)中的字节码反编译成 Java 源代码的形式,打印出类中的公共域、方法、构造函数和静态初始值。 javap 的用法...

    java字节码加密

    Java字节码加密是保护Java应用程序源代码安全的重要技术手段,主要是为了防止恶意用户逆向工程分析、篡改或盗取程序的核心逻辑。在Java中,字节码(Bytecode)是程序经过编译后的中间表示,可以直接由Java虚拟机...

    Java字节码转换工具—Retrotranslator

    Java字节码转换工具Retrotranslator是一个用于解决软件兼容性问题的实用工具,尤其是在Java版本升级带来的不兼容性上。随着Java技术的不断迭代,新版本的特性常常不能在旧版本的JDK环境下运行,而Retrotranslator的...

    Java 字节码概述

    当 JVM 得到一个 Java 字节码应用程序后,便为该代码中一个类的每一个方法创建一个栈框架,以保存该方法的状态信息。每个栈框架包括以下三类信息: * 局部变量:是被组织为一个以字长为单位、从 0 开始计数的数组。...

    class运行器v6(可以运行java字节码文件的工具 含代码)

    标题中的“class运行器v6”是一个用于执行Java字节码文件的应用程序,它允许用户在没有完整Java环境的情况下运行单个.class文件。这个工具可能是由开发者为了方便测试或教学目的而创建的,特别是对于那些不熟悉或者...

    java字节码编辑器

    例如,`aload_0`表示加载本地变量表中的第一个对象引用,`iconst_1`则表示将整数1压入操作数栈。 字节码编辑器的工作流程通常包括以下几个步骤: 1. **分析**:首先,编辑器会读取`.class`文件,解析其中的字节码...

    Java字节码优化框架

    Soot作为一个独立的工具,能够对Java字节码进行优化和检查,同时也为开发者提供了一个框架,以便于在字节码级别设计和实现优化策略。这个框架支持多种中间表示(Intermediate Representation,IR),使得代码分析和...

    java字节码文件查看工具,查看class文件

    总的来说,JD-GUI是一款强大的Java字节码查看工具,它的便捷性和功能强大性使得开发者可以更深入地理解并分析Java程序。无论是为了学习、调试还是逆向工程,掌握如何有效地使用此类工具都是Java开发者必备的技能之一...

    java字节码分析工具

    Java字节码分析工具,系统分析了java字节码文件,即java class类文件,对该文件中的各种成分以树的形式描述出来,只能针对未加密的class文件,一般由标准java编译器编译生成的class文件都未加密,该系统在vs2003下面...

    java字节码.docx

    魔数是 Java 字节码文件的标识符,它总是以 ca fe ba be 开头,表明该文件是一个 Java 字节码文件。 版本号(Version Number) Java 字节码文件的版本号主要包括主版本号和次版本号。主版本号表示 Java 字节码文件...

    Java字节码简单说明.docx

    Class 文件结构是 Java 字节码的载体,它是一种特殊的二进制文件格式,包含了 Java 虚拟机指令集和符号表以及若干其他辅助信息。Class 文件中存储了 Java 程序的定义信息,但它并不一定以磁盘文件的形式存在。 ...

    一个牛逼的 Java 字节码类库!(csdn)————程序.pdf

    本文主要介绍了一个强大的Java字节码处理类库——Javaassist。 Javaassist是一个开源库,允许开发者在运行时动态地修改或者创建Java类。它提供了一种简洁的API,使得程序员无需深入了解字节码的细节就能实现复杂的...

    Java 字节码简单说明.zip

    Java字节码是Java编程语言的一个重要特性,它在Java程序执行过程中扮演着核心角色。本文将深入探讨Java字节码的基本概念、作用以及其在Java虚拟机(JVM)中的运行机制。 Java字节码是一种低级的、平台无关的指令集...

    Java字节码(.class文件)格式详解((转载)

    Java字节码是Java程序在运行时被JVM(Java虚拟机)解释执行的一种中间语言。每个Java类都由一个`.class`文件表示,其中包含了编译后的字节码指令。`.class`文件的结构非常严谨,它不仅包含了类的信息,如类名、方法...

    从一个class文件深入理解Java字节码结构

    编译此源码(`javac Demo.java`)会产生一个名为`Demo.class`的字节码文件。 2. **字节码文件的查看** 使用文本编辑器(如Notepad++,配备HEX-Editor插件)可以打开`.class`文件,但直接查看十六进制数据难以理解...

    javassist, Java字节码工程工具包.zip

    javassist, Java字节码工程工具包 Java字节码工程工具包 版本 3版权所有( C ) 1999 -2017按 Shigeru Chiba,保留所有权利。Javassist ( Java编程助手) 使Java字节码操作简单。 它是一个类库,用于在Java中编辑字节码

    JAVA字节码操作库 BCEL

    BCEL(Byte Code Engineering Library)是Java开发的一个重要工具,主要用于处理Java字节码。它为开发者提供了一种深入理解与操作Java类文件的底层机制,允许分析、创建、修改和优化字节码。在软件工程中,BCEL在...

Global site tag (gtag.js) - Google Analytics