Java内存分配的分析
java的内存区域
- 寄存器
JVM内部虚拟寄存器,存取速度非常快,程序不可控制。 - 栈
保存局部变量的值,包括:- 1.用来保存基本数据类型的值;
- 2.保存类的实例,即堆区对象的引用(指针)。
- 3、也可以用来保存加载方法时的帧。
- 堆
用来存放动态产生的数据,比如new出来的对象。
注意创建出来的对象只包含属于各自的成员变量,并不包括成员方法。因为同一个类的对象拥有各自的成员变量,存储在各自的堆中,但是他们共享该类的方法,并不是每创建一个对象就把成员方法复制一次。 - 常量池
常量池是属于堆中的一块。
JVM为每个已加载的类型维护一个常量池,常量池就是这个类型用到的常量的一个有序集合。包括直接常量(基本类型,String)和对其他类型、方法、字段的符号引用。池中的数据和数组一样通过索引访问。 - 代码段
用来存放从硬盘上读取的源程序代码。 - 数据段
用来存放static定义的静态成员。

普通类型的变量和引用类型的变量在内存中操作的区别
无论是普通类型的变量还是引用类型的变量(俗称实例),都可以作为局部变量,他们都可以出现在栈中。只不过普通类型的变量在栈中直接保存它所对应的值,而引用类型的变量保存的是一个指向堆区的指针,通过这个指针,就可以找到这个实例在堆区对应的对象。因此,普通类型变量只在栈区占用一块内存,而引用类型变量要在栈区和堆区各占一块内存。
普通类型局部变量演示

date是基本类型,所以在栈中就直接存放了对应的值;而其他的都是实例变量,在栈中只是存放了对堆的指针,实际的值是在堆中的。

走到第一个change法,是对基本类型变量的修改。传入的值是date,JVM返现i是局部变量,将i放入到栈中,并将date的值赋给i。

修改i的值为1234.

change1执行完成后,JVM会立马释放i所占用的栈空间。
实例类型新建变量

JVM会给b在栈中创建空间,并将d1的指针地址赋值给b,由此b和d1都指向同一块对内存地址。

change2方法中又实例化了一个BirthDate对象,并且赋给b。在内部执行过程是:在堆区new了一个对象,并且把该对象的指针保存在栈中的b对应空间,此时实例b不再指向实例d1所指向的对象,但是实例d1所指向的对象并无变化,这样无法对d1造成任何影响。

由于b只是方法内变量,所以方法结束,则栈就将b回收了,但是堆中分配的空间还没释放,要等系统GC。
实例类型修改变量值


上面的操作和新建变量是一样的,b和d2都指向同一个堆的地址,最后b的修改其实也是对d2的修改。最后d2的值就是22了。
方法结束后b将会被回收。
总结
- 分清什么是实例什么是对象。Class a= new Class();此时a叫实例,而不能说a是对象。实例在栈中,对象在堆中,操作实例实际上是通过实例的指针间接操作对象。多个实例可以指向同一个对象。
- 栈中的数据和堆中的数据销毁并不是同步的。方法一旦结束,栈中的局部变量立即销毁,但是堆中对象不一定销毁。因为可能有其他变量也指向了这个对象,直到栈中没有变量指向堆中的对象时,它才销毁,而且还不是马上销毁,要等垃圾回收扫描时才可以被销毁。
- 以上的栈、堆、代码段、数据段等等都是相对于应用程序而言的。每一个应用程序都对应唯一的一个JVM实例,每一个JVM实例都有自己的内存区域,互不影响。并且这些内存区域是所有线程共享的。这里提到的栈和堆都是整体上的概念,这些堆栈还可以细分。
- 类的成员变量在不同对象中各不相同,都有自己的存储空间(成员变量在堆中的对象中)。而类的方法却是该类的所有对象共享的,只有一套,对象使用方法的时候方法才被压入栈,方法不使用则
不占用内存(不占用栈空间)。