1、堆栈概述
通俗来讲,栈与堆都是Java用来在RAM中存放数据的地方。堆主要用来存放new创建的对象和数组,栈主要是存储基本类型的变量和对象的引用,与C++不同,Java自动管理栈和堆,程序员不能直接地操作栈或堆。
Java的堆是一个运行时数据区,类的对象从中分配空间。这些对象通过new、newarray、anewarray和multianewarray等指令建立,它们不需要程序代码来显式的释放。堆是由GC(垃圾回收)来负责的,堆的优势是可以动态地分配内存大小,生存期也不必事先告诉编译器,因为它是在运行时动态分配内存的,Java的GC机制会自动收走这些不再使用的数据。但缺点是,由于要在运行时动态分配内存,存取速度较慢。
栈的优势是,存取速度比堆要快,仅次于寄存器,栈数据可以共享。但缺点是,存在栈中的数据大小与生存期必须是确定的,缺乏灵活性。栈中主要存放一些基本类型的变量(int, short, long, byte, float, double, boolean, char)和对象句柄。(注意:这里不包含String,String会在后边讲解。
假设我们定义一个list对象:
List list = new ArrayList();
上述语句在内存中执行的过程是:首先在堆中产生了一个ArrayList对象,然后在栈中定义一个特殊的变量list,让栈中这个变量的取值等于ArrayList对象在堆内存中的首地址,栈中的这个变量就成了该ArrayList对象的引用变量。 引用变量就相当于是该对象起的一个名称,以后就可以在程序中使用栈中的引用变量来访问堆中的对象。
2、堆栈中数据的释放
引用变量是普通的变量,定义时在栈中分配,引用变量在程序运行到其作用域之外后被释放。而数组和对象本身在堆中分配,即使程序运行到使用 new 产生数组或者对象的语句所在的代码块之外,数组和对象本身占据的内存不会被释放,数组和对象在没有引用变量指向它的时候,才变为垃圾,不能在被使用,但仍然占据内存空间不放,在随后的一个不确定的时间被GC收走(释放掉)。这也是 Java 比较占内存的原因。
实际上,栈中的变量指向堆内存中的变量,这就是Java中的指针!
3、栈的数据共享
栈有一个很重要的特殊性,就是存在栈中的数据可以共享。假设我们同时定义:
int a = 3; int b = 3;
编译器先处理int a = 3;首先它会在栈中创建一个变量为a的引用,然后查找栈中是否有3这个值,如果没找到,就将3存放进来,然后将a指向3。接着处理int b = 3;在创建完b的引用变量后,因为在栈中已经有3这个值,便将b直接指向3。这样,就出现了a与b同时均指向3的情况。
这时,如果再令a=4;那么编译器会重新搜索栈中是否有4值,如果没有,则将4存放进来,并令a指向4;如果已经有了,则直接将a指向这个地址。因此a值的改变不会影响到b的值。
要注意这种数据的共享与两个对象的引用同时指向一个对象的这种共享是不同的,因为这种情况a的修改并不会影响到b, 它是由编译器完成的,它有利于节省空间。而一个对象引用变量修改了这个对象的内部状态,会影响到另一个对象引用变量。
4、String类的处理方式
String是一个特殊的包装类数据。可以用:
String str = new String("abc"); String str = "abc";
两种的形式来创建,第一种是用new()来新建对象的,它会在存放于堆中。每调用一次就会创建一个新的对象。而第二种是先在栈中创建一个对String类的对象引用变量str,然后查找栈中有没有存放"abc",如果没有,则将"abc"存放进栈,并令str指向”abc”,如果已经有”abc” 则直接令str指向“abc”。
比较类里面的数值是否相等时,用equals()方法;当测试两个包装类的引用是否指向同一个对象时,用==,下面用例子说明上面的理论。
String str1 = "abc"; String str2 = "abc"; System.out.println(str1==str2); //true
可以看出str1和str2是指向同一个对象的。
String str1 =new String ("abc"); String str2 =new String ("abc"); System.out.println(str1==str2); // false
用new的方式是生成不同的对象。每一次生成一个。只要是用new()来新建对象的,都会在堆中创建,而且其字符串是单独存值的,即使与栈中的数据相同,也不会与栈中的数据共享。但像String str = "abc";这种场合下,其字符串值却是保存了一个指向存在栈中数据的引用!
因此用第二种方式创建多个”abc”字符串,在内存中其实只存在一个对象而已. 这种写法有利与节省内存空间. 同时它可以在一定程度上提高程序的运行速度,因为JVM会自动根据栈中数据的实际情况来决定是否有必要创建新对象。而对于String str = new String("abc");的代码,则一概在堆中创建新对象,而不管其字符串值是否相等,是否有必要创建新对象,从而加重了程序的负担。
另一方面, 要注意: 我们在使用诸如String str = "abc";的格式定义类时,总是想当然地认为,创建了String类的对象str。担心陷阱!对象可能并没有被创建!而可能只是指向一个先前已经创建的对象。只有通过new()方法才能保证每次都创建一个新的对象。
补充一点:
关于String str = "abc"的内部工作。Java内部将此语句转化为以下几个步骤:
(1)先定义一个名为str的对String类的对象引用变量:String str;
(2) 在栈中查找有没有存放值为"abc"的地址,如果没有,则开辟一个存放字面值为"abc"的地址,接着创建一个新的String类的对象o,并将o的字符 串值指向这个地址,而且在栈中这个地址旁边记下这个引用的对象o;如果已经有了值为"abc"的地址,则查找对象o,并返回o的地址。
(3)将str指向对象o的地址。
事实上,String类被设计成为不可改变(immutable)的类。如果你要改变其值,可以,但JVM在运行时根据新值悄悄创建了一个新对象,然后将这 个对象的地址返回给原来类的引用。这个创建过程虽说是完全自动进行的,但它毕竟占用了更多的时间。在对时间要求比较敏感的环境中,会带有一定的不良影响。所以操作字符串的时候,还是建议使用StringBuffer。
5、包装类一点特殊的地方
这里不是讲堆栈的问题,只是一个小知识点顺便记录下来。
请先看下边的例子:
Integer i1 = new Integer(3); Integer i2 = new Integer(3); System.out.println(i1 == i2);//false Integer i3 = new Integer(137); Integer i4 = new Integer(137); System.out.println(i3 == i4);//false Integer i5 = Integer.valueOf("3"); Integer i6 = Integer.valueOf("3"); System.out.println(i5 == i6);//true Integer i7 = Integer.valueOf("137"); Integer i8 = Integer.valueOf("137"); System.out.println(i7 == i8);//false Integer i9 = 3; Integer i10 = 3; System.out.println(i9 == i10);//true Integer i11 = 137; Integer i12 = 137; System.out.println(i11 == i12);//false
对于基本数据类型,如果在一个字节内,(即:-128 ~ 127)之间,一旦被包装成Integer对象的时候,就会被放到一个缓冲池中,当下次又需要将这个整数包装成Integer对象的时候,则会首先查找缓冲池中有没有这个对象,有的换就直接取出来。小整数装箱出来的都是同一个对象。这个结论仅仅适用于自动装箱操作,如果显式的进行装箱(即使用new的方式),则不合适。由于较小整数使用频率较高,此种操作节省了内存空间,这种模式叫做:享元模式-->flyweight
补充1:
简单通俗的讲,一个完整的Java程序运行过程会涉及以下内存区域:
寄存器:JVM内部虚拟寄存器,存取速度非常快,程序不可控制。
栈:保存局部变量的值,包括:1.用来保存基本数据类型的值;2.保存类的实例,即堆区对象的引用(指针)。也可以用来保存加载方法时的帧。
堆:用来存放动态产生的数据,比如new出来的对象。注意创建出来的对象只包含属于各自的成员变量,并不包括成员方法。因为同一个类的对象拥有各自的成员变量,存储在各自的堆中,但是他们共享该类的方法,并不是每创建一个对象就把成员方法复制一次。
常量池:JVM为每个已加载的类型维护一个常量池,常量池就是这个类型用到的常量的一个有序集合。包括直接常量(基本类型,String)和对其他类型、方法、字段的符号引用。池中的数据和数组一样通过索引访问。由于常量池包含了一个类型所有的对其他类型、方法、字段的符号引用,所以常量池在Java的动态链接中起了核心作用。常量池存在于堆中。
代码段:用来存放从硬盘上读取的源程序代码。
数据段:用来存放static定义的静态成员。
注:此文章部分内容来自互联网,具体出处不明。如不慎侵害的您的相关权益,请留言告知,本人将尽快删除相关内容。
本文链接:java中堆栈。转载请注明出处,谢谢。
相关推荐
描述: 文章介绍了使用Java编程语言实现了基于数组的Java堆栈,并实现了一些基本的堆栈方法。 标签: Java 数组 堆栈 Java数组堆栈的实现 Java数组堆栈的实现是通过创建一个名为ArrayStack的类,该类提供了堆栈的...
java堆栈信息dump文件
java dump 堆栈 dumpAnalyzer 分析,在日常工作中,经常会遇到,系统跑着跑着就会出现性能问题,CPU居高不下。这个时候我们就需要对系统的堆栈信息进行分析。这里就介绍如何使用IBM内存检测工具(dumpAnalyzer)。
在IT行业中,尤其是在Java应用程序的开发和调试过程中,理解并分析线程堆栈是非常关键的。线程堆栈反映了程序运行时的线程状态,帮助开发者定位和解决多线程问题,如死锁、资源竞争等。IBM为WebSphere Application ...
下面我们将详细讨论如何在Java中实现堆栈,并探讨其相关知识。 首先,我们可以通过自定义一个类来实现堆栈。基本的堆栈操作包括压栈(push)、弹栈(pop)、查看栈顶元素(peek)和检查堆栈是否为空(isEmpty)。...
* Java错误堆栈中的字符串可能会出现嵌套情况,如一个Java错误可能包含多个子错误。 改进的汉明距离算法 为了解决Java错误堆栈相似度计算的问题,本文提出了一种改进的汉明距离算法。该算法根据Java错误堆栈的特点...
Java堆栈分析是Java应用程序性能调优的重要环节,特别是在服务器环境中,当CPU资源占用过高或者出现内存泄露等问题时,分析Java堆栈能够帮助我们找出问题的根源。在本例中,我们将通过一系列步骤来理解如何进行Java...
每个`StackTraceElement`对象表示堆栈中的一个帧,包含了方法名、文件名、行号等信息。通过遍历这个`Map`,我们可以打印出任意线程的堆栈信息。 ```java private void PrintCallStack() { java.util.Map, ...
Java堆栈是一个重要的内存区域,它是Java虚拟机(JVM)的一部分,主要负责管理方法的执行。在这个"java 堆栈的演示程序"中,我们可能会深入理解堆栈的工作原理以及它在运行jsp程序时的角色。源代码设计将帮助我们...
当 JVM 激活一个方法时,它从类信息数据得到此方法的局部变量区和操作数堆栈的大小,并据此分配大小合适的堆栈帧压入 Java 堆栈中。 在局部变量区中,类型为 int、float、reference 和 returnAddress 的值占据一项...
本文将深入探讨Java堆栈的概念、工作原理以及它们在程序执行中的角色。 1. 堆(Heap) - 堆是Java内存模型中的主要部分,主要用于存储对象实例。所有的类实例和数组都在堆中分配内存。 - 堆内存是动态分配的,...
java线程堆栈分析工具jca466.jar;堆内存分析工具Memory Analyzer;分析内存泄露产生的javacore文件,以便于定位blocked线程
Java 把内存分成两种,一种叫做栈...当在一段代码块中定义一个变量时,java 就在栈中 为这个变量分配内存空间,当超过变量的作用域后,java 会自动释放 掉为该变量分配的内存空间,该内存空间可以立刻被另作他用。
在这个Java计算器的实现中,我们首先将输入的表达式分解为一个个字符,然后使用堆栈来处理这些字符。遇到数字时,我们将它们压入堆栈作为操作数;遇到操作符时,我们比较其与栈顶操作符的优先级,如果当前操作符...
在这个“java泛型容器堆栈代码”中,我们将会探讨Java如何利用泛型来创建高效且类型安全的容器,特别是堆栈数据结构。 堆栈是一种基于“后进先出”(LIFO)原则的数据结构,常用于实现函数调用栈、内存管理等场景。...
Java堆栈内存分析是Java编程中的重要概念,它关乎程序的性能优化和内存泄漏的预防。堆和栈是Java内存管理的两个主要区域,它们各自承担着不同的职责。本笔记将深入探讨这两个区域的工作原理以及如何进行有效的分析。...
堆栈实现的java 计算器 + - * / % ()
Java线程堆栈是一种强大的诊断工具,能够帮助开发者快速定位多线程应用程序中的问题。通过分析线程堆栈,可以找到系统中各种问题的根源,如系统无缘无故CPU过高、系统挂起、系统运行越来越慢、性能瓶颈、线程死锁、...
### Java堆栈的区别详解 #### 一、预备知识—程序的内存分配 程序在运行时,根据不同的数据类型和用途,会被分配到不同的内存区域。这些区域包括: 1. **栈区(Stack)**:这部分内存由编译器自动管理,主要用于...
// size 方法,返回堆栈中的元素个数 this.size = function () { var count = 0; var current = this.head.next; while (current !== null) { count++; current = current.next; } return count; } } ``` ...