java bytecode - nessuna istruzione ipush


Come parte di uno dei miei progetti scolastici ho bisogno di scavare nel bytecode java. Ho iniziato a scrivere programmi semplici e usare l'utilità javap per visualizzare bytecode generato e ho una domanda riguardante le istruzioni *ipush.

Quando visualizzo il bytecode di questo codice:

public class Main{
  public static void main(String []args){
    int a;
    a=5;
    a=a+32765;
  }
}

Sto ottenendo

public class Main
  SourceFile: "Main.java"
  minor version: 0
  major version: 51
  flags: ACC_PUBLIC, ACC_SUPER
Constant pool:
  #1 = Methodref          #3.#12         //  java/lang/Object."<init>":()V
  #2 = Class              #13            //  Main
  #3 = Class              #14            //  java/lang/Object
  #4 = Utf8               <init>
  #5 = Utf8               ()V
  #6 = Utf8               Code
  #7 = Utf8               LineNumberTable
  #8 = Utf8               main
  #9 = Utf8               ([Ljava/lang/String;)V
  #10 = Utf8               SourceFile
  #11 = Utf8               Main.java
  #12 = NameAndType        #4:#5          //  "<init>":()V
  #13 = Utf8               Main
  #14 = Utf8               java/lang/Object
{
  public Main();
flags: ACC_PUBLIC
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[]);
flags: ACC_PUBLIC, ACC_STATIC
Code:
  stack=2, locals=2, args_size=1
    0: iconst_5      
    1: istore_1      
    2: iload_1       
    3: sipush        32765
    6: iadd          
    7: istore_1      
    8: return        
  LineNumberTable:
    line 4: 0
    line 5: 2
    line 6: 8
}

E quando cambio 32765 per 32769 che non è nell'intervallo di short sto ottenendo

public class Main
  SourceFile: "Main.java"
  minor version: 0
  major version: 51
  flags: ACC_PUBLIC, ACC_SUPER
Constant pool:
  #1 = Methodref          #4.#13         //  java/lang/Object."<init>":()V
  #2 = Integer            32769
  #3 = Class              #14            //  Main
  #4 = Class              #15            //  java/lang/Object
  #5 = Utf8               <init>
  #6 = Utf8               ()V
  #7 = Utf8               Code
  #8 = Utf8               LineNumberTable
  #9 = Utf8               main
  #10 = Utf8               ([Ljava/lang/String;)V
  #11 = Utf8               SourceFile
  #12 = Utf8               Main.java
  #13 = NameAndType        #5:#6          //  "<init>":()V
  #14 = Utf8               Main
  #15 = Utf8               java/lang/Object
{
  public Main();
flags: ACC_PUBLIC
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[]);
flags: ACC_PUBLIC, ACC_STATIC
Code:
  stack=2, locals=2, args_size=1
    0: iconst_5      
    1: istore_1      
    2: iload_1       
    3: ldc           #2                  // int 32769
    5: iadd          
    6: istore_1      
    7: return        
  LineNumberTable:
    line 4: 0
    line 5: 2
    line 6: 7
}

Quindi ora il numero è memorizzato nel pool costante. So che non c'è istruzione ipush che potrei usare per spingere la costante intera allo stack, ma mi chiedo perché non ci sia tale istruzione?

Author: Andna, 2013-06-02

1 answers

Sulla base di un semplice design CISC, hanno cercato di rendere ogni istruzione il più breve possibile (in byte) Le istruzioni che utilizzano costanti a 32 o 64 bit sono più facili/brevi per riferirsi a una tabella costante.

 3
Author: Peter Lawrey, 2013-06-02 10:35:08