JVM 符号引用和直接引用

符号引用

符号引用以一组符号来描述所引用的目标。符号引用可以是任何形式的字面量,只要使用时能无歧义地定位到目标即可。符号引用和虚拟机的布局无关。
个人理解:在编译的时候一个每个java类都会被编译成一个class文件,但在编译的时候虚拟机并不知道所引用类的地址,多以就用符号引用来代替,而在这个解析阶段就是为了把这个符号引用转化成为真正的地址的阶段
 
直接引用
直接引用就是直接指向目标的指针、相对偏移量或一个间接定位到目标的句柄。
 
字面量
字面量表示如何表达这个值,一般除去表达式,给变量赋值时,等号右边都可以认为是字面量
字面量分为字符串字面量(string literal )、数组字面量(array literal)和对象字面量(object literal),另外还有函数字面量(function literal)。
示例:var test="hello world!";
"hello world!" 就是字符串字面量,test 是变量名。
 
例如,在Class文件中它以CONSTANT_Class_info、CONSTANT_Fieldref_info、CONSTANT_Methodref_info等类型的常量出现。
比如org.simple.People类引用了org.simple.Language类,在编译时People类并不知道Language类的实际内存地址,因此只能使用符号org.simple.Language(假设是这个,当然实际中是由类似于CONSTANT_Class_info的常量来表示的)来表示Language类的地址。各种虚拟机实现的内存布局可能有所不同,但是它们能接受的符号引用都是一致的,因为符号引用的字面量形式明确定义在Java虚拟机规范的Class文件格式中。
 
 
下面就通过下面示例来加深符号引用与直接引用的理解:
public class RefTest {
    
    public void method() {
        int i = 10;
        String str = "test";
    }
}

使用javap -v 反编译后

  Last modified 2020-4-17; size 423 bytes
  MD5 checksum 8b46cded3501cd81c9cef4a1d03dc398
  Compiled from "RefTest.java"
public class com.classloader.RefTest
  minor version: 0
  major version: 52
  flags: ACC_PUBLIC, ACC_SUPER
Constant pool:
   #1 = Class              #2             // com/classloader/RefTest
   #2 = Utf8               com/classloader/RefTest
   #3 = Class              #4             // java/lang/Object
   #4 = Utf8               java/lang/Object
   #5 = Utf8               <init>
   #6 = Utf8               ()V
   #7 = Utf8               Code
   #8 = Methodref          #3.#9          // java/lang/Object."<init>":()V
   #9 = NameAndType        #5:#6          // "<init>":()V
  #10 = Utf8               LineNumberTable
  #11 = Utf8               LocalVariableTable
  #12 = Utf8               this
  #13 = Utf8               Lcom/classloader/RefTest;
  #14 = Utf8               method
  #15 = String             #16            // test
  #16 = Utf8               test
  #17 = Utf8               i
  #18 = Utf8               I
  #19 = Utf8               str
  #20 = Utf8               Ljava/lang/String;
  #21 = Utf8               SourceFile
  #22 = Utf8               RefTest.java
{
  public com.classloader.RefTest();
    descriptor: ()V
    flags: ACC_PUBLIC
    Code:
      stack=1, locals=1, args_size=1
         0: aload_0
         1: invokespecial #8                  // Method java/lang/Object."<init>":()V
         4: return
      LineNumberTable:
        line 3: 0
      LocalVariableTable:
        Start  Length  Slot  Name   Signature
            0       5     0  this   Lcom/classloader/RefTest;

  public void method();
    descriptor: ()V
    flags: ACC_PUBLIC
    Code:
      stack=1, locals=3, args_size=1
         0: bipush        10
         2: istore_1
         3: ldc           #15                 // String test
         5: astore_2
         6: return
      LineNumberTable:
        line 6: 0
        line 7: 3
        line 8: 6
      LocalVariableTable:
        Start  Length  Slot  Name   Signature
            0       7     0  this   Lcom/classloader/RefTest;
            3       4     1     i   I
            6       1     2   str   Ljava/lang/String;
}
SourceFile: "RefTest.java"

 

 

 

posted @ 2020-04-20 03:06  cao_xiaobo  阅读(798)  评论(0编辑  收藏  举报