Java对象内存分配全解:从new Student()到this关键字,一张图看懂对象在内存中的完整生命周期
当你写下这行代码时Student stu new Student();你有没有想过这短短一行代码在内存中到底发生了什么对象存在哪里stu变量里存的到底是什么为什么方法里修改对象的属性外面的对象也跟着变了this到底指向谁如果你对这些问题似懂非懂那今天这篇文章就是为你准备的。我们将从内存角度一步步拆解Java对象的完整生命周期彻底搞懂面向对象底层原理。一、Java虚拟机三大内存区域快速回顾在开始之前先快速回顾一下Java虚拟机的三个核心内存区域区域作用特点栈内存存储方法调用和局部变量方法调用入栈执行完毕出栈堆内存存储new出来的对象、数组所有new创建的空间都在这里方法区存储类的字节码信息.class文件类加载时进入虚拟机关闭才释放重要方法区的字节码信息不会随方法执行完毕而消失它一直存在直到关闭虚拟机。二、创建对象的完整七步流程我们以最常见的代码为例Student stu new Student();这行代码在内存中经历了七个步骤第一步加载class字节码文件程序启动后Student.class文件被加载到方法区。类的所有信息属性、方法都存在这里等待被使用。第二步声明局部变量在栈内存中为变量stu开辟空间。这个变量将来要存储对象的地址。第三步在堆中开辟空间因为new关键字在堆内存中开辟一块空间。假设这块空间的地址是0x0011。这块空间就是我们平时说的“对象”。对象空间里存什么所有的成员属性name、age等所有成员方法在方法区中的地址方便调用时快速找到第四步属性默认初始化给成员属性赋予默认值int→0double→0.0boolean→false引用类型String、数组等→null第五步属性显式初始化如果定义成员变量时直接赋值了比如int age 18;就在这一步把默认值替换成你写的值。第六步构造方法初始化调用构造方法给成员变量赋值。如果用的是有参构造这一步就会把参数值赋给属性如果是无参构造这一步可能什么也不做。注意构造方法只是创建对象过程中的一步不是全部。第七步将地址赋值给变量将堆中对象的地址0x0011赋值给栈中的变量stu。从此stu这个变量就“指向”了堆中的对象。理解这一点很重要stu不是对象本身它只是存了对象的地址。就像你手里的门牌号不是房子本身但能帮你找到房子。三、对象的访问流程访问属性stu.name 小诗诗;从栈中找到stu拿到它存的地址0x0011通过地址找到堆中的对象修改或读取对应属性的值调用方法stu.study();从stu拿到地址0x0011找到堆中的对象对象里存了study方法在方法区的地址跳转到方法区执行study方法的字节码study方法入栈执行四、对象的生命周期与回收规则规则一方法出栈局部变量消失main方法执行完毕出栈后栈中的stu变量也随之消失。规则二没有变量引用的对象被回收当没有任何变量指向堆中的对象时这个对象就会被JVM垃圾回收器回收。因为已经找不到它了留着也没用。规则三方法区的字节码直到虚拟机关闭才消失类加载一次就够了字节码信息会一直存在直到关闭IDEA或关闭Java程序。五、对象作为方法参数传递的内存原理这是面试中经常问到的问题也是理解“为什么方法内修改对象会影响原对象”的关键。来看这段代码public class MemoryDemo { public static void main(String[] args) { Student stu new Student(); stu.name 小诗诗; stu.age 23; change(stu); System.out.println(stu.name ... stu.age); // 输出小丹丹...19 } public static void change(Student stu) { stu.name 小丹丹; stu.age 19; } }内存执行过程创建对象main方法中stu变量存着对象的地址0x0011对象属性是“小诗诗”和23。调用change方法change方法入栈形参stu也在栈中开辟空间。调用时把实参stu中存的地址0x0011复制一份传给形参stu。两个变量指向同一个对象现在栈中有两个stu变量一个是main的一个是change的它们存的地址都是0x0011指向同一个堆对象。修改对象change方法中通过形参stu找到堆中对象把name改成“小丹丹”age改成19。这是直接修改堆中的对象。方法出栈change方法执行完毕形参stu消失。但堆中的对象已经被修改了。回到main方法main的stu变量还在它存的地址还是0x0011访问到的就是修改后的值。核心结论Java中传递对象参数时传递的是对象的内存地址不是对象本身。传递地址后多个变量指向同一个对象。任何一个变量修改对象其他变量访问到的都是修改后的值。六、this关键字的内存本质this是Java中一个非常关键的关键字它的作用有两个区分同名变量当成员变量和局部变量同名时用this.访问成员变量代表当前对象哪个对象调用了方法this就指向哪个对象内存理解this就是调用者的地址public class Student { String name; int age; public void study() { int age 10; // 局部变量 System.out.println(age); // 10就近原则 System.out.println(this.age); // 成员变量的值 } }测试代码public class Test { public static void main(String[] args) { Student s1 new Student(小诗诗, 18); Student s2 new Student(小丹丹, 19); s1.study(); // 输出10 18 s2.study(); // 输出10 19 } }内存执行过程创建两个对象堆中有两个独立的对象地址分别是0x0011和0x0022。s1调用study方法study方法入栈方法内部自动记录调用者的地址。这个地址就是this此时this 0x0011。打印age不带this的age是局部变量输出10。打印this.agethis.age就是通过this存的地址0x0011找到s1对象的age输出18。s2调用study方法study方法再次入栈这次this 0x0022找到s2对象的age输出19。核心结论this的本质就是当前方法调用者的内存地址。哪个对象调用了方法this就指向哪个对象同一个类的多个对象共享方法区的同一份方法字节码依靠this区分“是谁在调用”七、总结一张图看懂对象内存分配内存区域存储内容生命周期栈内存局部变量存地址方法执行完就消失堆内存对象实例属性方法地址无引用时被回收方法区类字节码虚拟机关闭才消失核心理解变量存的是地址不是对象本身对象传参传的是地址所以方法内修改会影响原对象this就是调用者的地址用来区分“谁在调用”写在最后面向对象是Java的灵魂而理解内存中的对象是掌握面向对象的必经之路。
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2437919.html
如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!