读懂 Java 字节码(1)

java-logo.jpg

在介绍新语言都会对比 java 来讲,以贬低 java 抬高自己方式来介绍语言的一些方便之处。Java 的确存在许多让人头痛的问题。而且由于不断吸收变得庞大、臃肿。通过添加新特性来让自己变得强大并不是坏事,但是变得越来越复杂也是其副作用。从而失去自己的特点。这也是 rob pike 坚持己见,对 go 不做应适修改的原因。rob bike 所倡导的简单也是值得我们对于一些当下流行语言反思。要做到复杂可能并不难,不过要做的简单可能需要多花一些精力。


现在许多新兴的语言都可以运行在 JVM 上,这是因为他们都可以编译为字节码。只要你的语言最终编译成符合 JVM 规范的字节码就可以运行在 JVM 上。这样我们就也可以写出自己语言来运行在 JVM 虚拟机上。

JVM java 的虚拟机是基于栈结构。都会在栈中产生一个帧。

package com.zidea.test;

public class Demo {
    private int a = 1;

    public Demo() {
    }

    public int getA() {
        return this.a;
    }

    public void setA(int a) {
        this.a = a;
    }
}

Demo.class
  Last modified 2019-4-3; size 466 bytes
  MD5 checksum 49bf9f0b6522624b609713b4f94a3cb5
  Compiled from "Demo.java"
public class com.zidea.test.Demo
  minor version: 0
  major version: 52
  flags: ACC_PUBLIC, ACC_SUPER
Constant pool:
   #1 = Methodref          #4.#20         // java/lang/Object."<init>":()V
   #2 = Fieldref           #3.#21         // com/zidea/test/Demo.a:I
   #3 = Class              #22            // com/zidea/test/Demo
   #4 = Class              #23            // java/lang/Object
   #5 = Utf8               a
   #6 = Utf8               I
   #7 = Utf8               <init>
   #8 = Utf8               ()V
   #9 = Utf8               Code
  #10 = Utf8               LineNumberTable
  #11 = Utf8               LocalVariableTable
  #12 = Utf8               this
  #13 = Utf8               Lcom/zidea/test/Demo;
  #14 = Utf8               getA
  #15 = Utf8               ()I
  #16 = Utf8               setA
  #17 = Utf8               (I)V
  #18 = Utf8               SourceFile
  #19 = Utf8               Demo.java
  #20 = NameAndType        #7:#8          // "<init>":()V
  #21 = NameAndType        #5:#6          // a:I
  #22 = Utf8               com/zidea/test/Demo
  #23 = Utf8               java/lang/Object
{
  private int a;
    descriptor: I
    flags: ACC_PRIVATE

  public com.zidea.test.Demo();
    descriptor: ()V
    flags: ACC_PUBLIC
    Code:
      stack=2, locals=1, args_size=1
         0: aload_0
         1: invokespecial #1                  // Method java/lang/Object."<init>":()V
         4: aload_0
         5: iconst_1
         6: putfield      #2                  // Field a:I
         9: return
      LineNumberTable:
        line 3: 0
        line 4: 4
      LocalVariableTable:
        Start  Length  Slot  Name   Signature
            0      10     0  this   Lcom/zidea/test/Demo;

  public int getA();
    descriptor: ()I
    flags: ACC_PUBLIC
    Code:
      stack=1, locals=1, args_size=1
         0: aload_0
         1: getfield      #2                  // Field a:I
         4: ireturn
      LineNumberTable:
        line 7: 0
      LocalVariableTable:
        Start  Length  Slot  Name   Signature
            0       5     0  this   Lcom/zidea/test/Demo;

  public void setA(int);
    descriptor: (I)V
    flags: ACC_PUBLIC
    Code:
      stack=2, locals=2, args_size=2
         0: aload_0
         1: iload_1
         2: putfield      #2                  // Field a:I
         5: return
      LineNumberTable:
        line 11: 0
        line 12: 5
      LocalVariableTable:
        Start  Length  Slot  Name   Signature
            0       6     0  this   Lcom/zidea/test/Demo;
            0       6     1     a   I
}
SourceFile: "Demo.java"

我们先简单浏览一遍
private int a = 1; 中 descriptor 为描述符,flags 标识符,其中包含修饰变量的修饰符信息。

  • descriptor: I
  • flags: ACC_PRIVATE

private static int b;如果是这样的定义一个静态变量,标识符就多了一个表示静态变量 ACC_STATIC
descriptor: I
flags: ACC_PRIVATE, ACC_STATIC

0: aload_0
1: invokespecial #1                  // Method java/lang/Object."<init>":()V
4: aload_0
5: iconst_1
6: putfield      #2                  // Field a:I
9: return

这是一个构造函数的 code

  • aload_0 加载 this

可以用一些工具来查看字节码文件

  • aload_0: 从局部变量的相应位置装载一个对象引用到操作数栈的栈顶。尽管示例代码中并不包含构造函数,我们知道 JVM 会类添加一个默认的构造函数。但是类的变量的初始化代码实际会由编译器构建构建默认构造函数中执行。

  • invokespecial: 用于调用实例的初始化方法,包括私有方法以及当前类的父类方法。同样属于一组以不同方式来调用方法的操作码,这些操作码包括invokedynamic、invokeinterface、invokespecial、invokestatic和invokevirtual。invokespecial是用于调用父类构造方法的指令,例如java.lang.Object 的构造方法。

// Method java/lang/Object."<init>":()V

  • putfield: 从运行时的常量池中取一个指向成员变量的引用,这个成员变量的值以及其对应的对象都会从操作数栈中弹出,本例中的成员变量即为 a。例子中,aload_0 首先向操作数栈中添加了对象,然后 iconst_1 向操作数栈中添加了 1 这个值,最后putfield从栈中弹出了这两个值,最终,这个对象的a的值被设置为1。

如果我们将 a 赋值为 100 即 private int a = 100;

 0: aload_0
  1: invokespecial #1                  // Method java/lang/Object."<init>":()V
  4: aload_0
  5: bipush        100
  7: putfield      #2                  // Field a:I
10: return

iconst_1 就变为 bipush 100

java_bean.jpg
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 158,847评论 4 362
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,208评论 1 292
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,587评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,942评论 0 205
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,332评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,587评论 1 218
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,853评论 2 312
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,568评论 0 198
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,273评论 1 242
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,542评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,033评论 1 260
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,373评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,031评论 3 236
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,073评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,830评论 0 195
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,628评论 2 274
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,537评论 2 269

推荐阅读更多精彩内容

  • Java byte code 的学习意义 为啥要学java bytecode,这就跟你问我已经会python了为...
    shanggl阅读 1,595评论 0 3
  • 这篇文章解释了Java 虚拟机(JVM)的内部架构。下图显示了遵守Java SE 7 规范的典型的 JVM 核心内...
    饮墨飨书阅读 615评论 0 1
  • 即便对那些有经验的Java开发人员来说,阅读已编译的Java字节码也很乏味。为什么我们首先需要了解这种底层的东西?...
    怪物小姐姐阅读 570评论 0 8
  • 每天我们都会遇到形形色色的路人,比如上下班途中的陌生人、公交地铁里的同路人、吃饭时的邻桌人、打听问路的有缘人、发呆...
    王美淇阅读 302评论 0 0
  • 若现在就感到失望无力,未来那么远你该怎么扛。别把人生想的太难。走过生命的逆旅,人世沧桑,谁都会彷徨,会忧伤,会有苦...
    得了急阅读 252评论 0 0