JVM的符号引用和直接引用是什么

解析一:
符号引用就是一个类中(当然不仅是类,还包括类的其他部分,比如方法,字段等),引入了其他的类,可是JVM并不知道引入的其他类在哪里,所以就用唯一符号来代替,等到类加载器去解析的时候,就把符号引用找到那个引用类的地址,这个地址也就是直接引用。

解析二:
1.符号引用(Symbolic References):

符号引用以一组符号来描述所引用的目标,符号可以是任何形式的字面量,只要使用时能够无歧义的定位到目标即可。例如,在Class文件中它以CONSTANT_Class_info、CONSTANT_Fieldref_info、CONSTANT_Methodref_info等类型的常量出现。符号引用与虚拟机的内存布局无关,引用的目标并不一定加载到内存中。在Java中,一个java类将会编译成一个class文件。在编译时,java类并不知道所引用的类的实际地址,因此只能使用符号引用来代替。比如org.simple.People类引用了org.simple.Language类,在编译时People类并不知道Language类的实际内存地址,因此只能使用符号org.simple.Language(假设是这个,当然实际中是由类似于CONSTANT_Class_info的常量来表示的)来表示Language类的地址。各种虚拟机实现的内存布局可能有所不同,但是它们能接受的符号引用都是一致的,因为符号引用的字面量形式明确定义在Java虚拟机规范的Class文件格式中。

2.直接引用:
直接引用可以是
(1)直接指向目标的指针(比如,指向“类型”【Class对象】、类变量、类方法的直接引用可能是指向方法区的指针)
(2)相对偏移量(比如,指向实例变量、实例方法的直接引用都是偏移量)
(3)一个能间接定位到目标的句柄
直接引用是和虚拟机的布局相关的,同一个符号引用在不同的虚拟机实例上翻译出来的直接引用一般不会相同。如果有了直接引用,那引用的目标必定已经被加载入内存中了。

解析三:
符号引用即用**(用字符串符号的形式)**来表示引用,其实被引用的类、方法或者变量还没有被加载到内存中。而直接引用则是有具体引用地址的指针,被引用的类、方法或者变量已经被加载到内存中。以变量举个例子:
在这里插入图片描述
符号引用要转换成直接引用才有效,这也说明直接引用的效率要比符号引用高。那为什么要用符号引用呢?这是因为类加载之前,javac会将源代码编译成.class文件,这个时候javac是不知道被编译的类中所引用的类、方法或者变量他们的引用地址在哪里,所以只能用符号引用来表示,当然,符号引用是要遵循java虚拟机规范的。还有一种情况需要用符号引用,就例如前文举得变量的符号引用的例子,是为了逻辑清晰和代码的可读性。
javac在编译一个A类时,如果A类应用了B类,那么javac到类路径下找B类,如果找到了,就把A类编译成A.class文件,这时A.class文件中有个B类的符号引用(字符串形式),这个可以自己写两个A类和B类,A类引用B类,编译A类后,使用JDK自带反编译工具,反编译出代码看

我们都知道,类加载过程分为加载—>验证—>准备—>解析—>初始化这 5个阶段,符号引用转换为直接引用就发生在解析阶段,解析阶段可能在初始化前,也可能在初始化之后。

学习张龙JVM

  • 58
    点赞
  • 155
    收藏
    觉得还不错? 一键收藏
  • 8
    评论
JVM(Java虚拟机)的内存模型是指在JVM中,Java程序运行时所使用的内存结构和管理方式。它包括以下几个部分: 1. 程序计数器(Program Counter):用于记录当前线程执行的字节码指令的地址。 2. Java虚拟机栈(JVM Stack):每个线程在执行Java方法时,会创建一个对应的栈帧(Stack Frame)。栈帧包含了方法的局部变量表、操作数栈、动态链接、方法出口等信息。 3. 本地方法栈(Native Method Stack):用于执行本地方法(Native Method)的栈。 4. 堆(Heap):用于存储对象实例和数组。堆是Java程序中最大的一块内存,被所有线程共享。JVM通过垃圾回收器(Garbage Collector)自动管理堆内存的分配和释放。 5. 方法区(Method Area):用于存储已加载的类信息、常量、静态变量、即时编译器编译后的代码等。方法区也被称为永久代(Permanent Generation),但在JDK 8及以后的版本中,永久代被元空间(Metaspace)所取代。 6. 运行时常量池(Runtime Constant Pool):每个类或接口在编译后都会生成一个运行时常量池,用于存放编译器生成的字面量和符号引用。 7. 直接内存(Direct Memory):JVM中的直接内存不是由JVM管理的,而是通过操作系统的本地IO直接分配的内存,一般用于NIO(New Input/Output)操作。 这些内存区域共同组成了JVM的内存模型,其结构和使用方式对于Java程序的运行和性能有着重要的影响。
评论 8
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值