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"