您的位置:首页 > 编程语言 > Java开发

java 栈 和 堆 的区别

2016-04-20 22:30 253 查看
转载:

一、java栈

Java 栈总是和线程关联在一起,每当创建一个线程时,JVM就会为这个线程创建一个对应的Java栈,在这个Java栈中又会含有多个栈(Frames),这些栈帧是与每个方法关联起来的,每个运行一个方法就创建一个栈帧会含有一些“内部变量"(在方法内定义的变量)、"操作栈"和"方法返回值"等信息。

二、堆

堆是存储Java 对象的地方,它是JVM管理 Java 对象的核心存储区域,堆是 Java 程序员最应该关心的,因为它是我们的应用程度与内存关系最密切的存储区域。

每一个存储在堆中的Java 对象都会是这个对象的类的副本,它会复制包括继承自它父类的所有非静态属性。

堆是被所有 Java 线程所共享的,所以对它的访问需要注意同步问题,方法和对应的属性都需要保证一致性。

内存可以分为3个区:方法区(又叫静态区method)、栈(stack)和堆(heap)

1.方法区

存放所有的class(即类中的函数也存放于此区)和静态变量,方法区存放的是整个程序中唯一的元素,如class和static变量。

可以被所有的线程共享,这一点和堆heap一样。

2.栈内存:

存储的都是局部变量。

而且变量所属的作用域一旦结束,该变量就自动释放。

3.堆内存

存储的都是实体,数组和对象(数组就是对象),每一个实体都有首地址值。

凡是new出来的都在堆内存,并都将进行默认的初始化。

不需要手动释放、自动垃圾回收。

?
堆内存中的实体数据地址都存储在栈变量中(即引用),以便能够高速访问。引用丢失后将被GC(垃圾回收器)列入可回收“名单”中,即变为垃圾,等待自动回收(区别于C++,C++中new出来的数组需要Delete,类的话需要析构函数进行析构才能释放内存)。

关于内存地址:





只需关注最左边的两个字符:

“[”表示这个实体是数组型实体

I表示int F表示float

java中变量在内存中的分配

1、类变量(static修饰的变量):在程序加载时系统就为它在堆中开辟了内存,堆中的内存地址存放于栈以便于高速访问。静态变量的生命周期--一直持续到整个"系统"关闭

2、实例变量:当你使用java关键字new的时候,系统在堆中开辟并不一定是连续的 空间分配给变量(比如说类实例),然后根据零散的堆内存地址,通过哈希算法换算为一长串数字以表征这个变量在堆中的"物理位置"。 实例变量的生命周期--当实例变量的引用丢失后,将被GC(垃圾回收器)列入可回收“名单”中,但并不是马上就释放堆中内存

3、局部变量:局部变量,由声明在某方法,或某代码段里(比如for循环),执行到它的时候在栈中开辟内存,当局部变量一但脱离作用域,内存立即释放

Java 把内存划分成两种:一种是栈内存,另一种是堆内存。在函数中定义的一些基本类型的变量和对象的引用变量都是在函数的栈内存中分配,当在一段代码块定义一个 变量时,Java 就在栈中为这个变量分配内存空间,当超过变量的作用域后,Java 会自动释放掉为该变量分配的内存空间,该内存空间可以立即被另作它用。

  堆内存用来存放由 new 创建的对象和数组,在堆中分配的内存,由 Java 虚拟机的自动垃圾回收器来管理。在堆中产生了一个数组或者对象之后,还可以在栈中定义一个特殊的变量,让栈中的这个变量的取值等于数组或对象在堆内存中的 首地址,栈中的这个变量就成了数组或对象的引用变量,以后就可以在程序中使用栈中的引用变量来访问堆中的数组或者对象,引用变量就相当于是为数组或者对象 起的一个名称。引用变量是普通的变量,定义时在栈中分配,引用变量在程序运行到其作用域之外后被释放。而数组和对象本身在堆中分配,即使程序运行到使用 new 产生数组或者对象的语句所在的代码块之外,数组和对象本身占据的内存不会被释放,数组和对象在没有引用变量指向它的时候,才变为垃圾,不能在被使用,但仍 然占据内存空间不放,在随后的一个不确定的时间被垃圾回收器收走(释放掉)。

  这也是 Java 比较占内存的原因,实际上,栈中的变量指向堆内存中的变量,这就是 Java 中的指针!
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: