`

Java变量以及内存分配(非常重要)

    博客分类:
  • Java
阅读更多
 

不知道是第几次看thinking in java了。不是的翻翻总有新的收获。

堆栈

静态存储区域

一个由C/C++编译的程序占用的内存分为以下几个部分

1、栈区(stack)— 由编译器自动分配释放 ,存放函数的参数值,局部变量的值等。其操作方式类似于数据结构中的栈。

2、堆区(heap)— 由程序员分配释放, 若程序员不释放,程序结束时可能由OS回收 。注意它与数据结构中的堆是两回事,分配方式倒是类似于链表

3、全局区(静态区)(static)— 全局变量和静态变量的存储是放在一块的,初始化的全局变量和静态变量在一块区域, 未初始化的全局变量和未初始化的静态变量在相邻的另一块区域。程序结束后由系统释放。

4、文字常量区 — 常量字符串就是放在这里的,程序结束后由系统释放 。

5、程序代码区 — 存放函数体的二进制代码。

 

Java中保存地址:

寄存器register 这是速度最快的地方 数据位于和其他所有方式都不同的一个地方 处理器的内部 不过 寄存器的数量十分有限 所以寄存器是根据需要由编译器分配 我们对此没有直接的控制权 也不可能在自己的程序里找到寄存器存在的任何迹象。

JVM的寄存器用来存放当前系统状态。然而,基于移植性要求,JVM拥有的寄存器数目不能过多。否则,对于任何本身的寄存器个数小于JVM的移植目标机,要用常规存储来模拟高速寄存器,是比较困难的。同时JVM是基于栈(Stack)的,这也使得它拥有的寄存器较少。

JVM的寄存器包括下面四个:

(1)PC程序计数寄存器
(2)optop操作数栈栈顶地址寄存器。
(3)frame当前执行环境地址寄存器。
(4)vars局部变量首地址寄存器。

这些寄存器长度均为32位。其中PC用来记录程序执行步骤,其余optop,frame,vars都存放JVM栈中对应地址,用来快速获取当前执行所需的信息。

堆栈stack 堆栈位于常规 RAM 随机访问存储器 内 但可通过它的 堆栈指针 获得处理器的直接支持 堆栈指针若向下移 会创建新的内存 若向上移 则会释放那些内存这是一种特别快 特别有效的数据保存方式 仅次于寄存器 创建程序时 Java编译器必须准确地知道堆栈内保存的所有数据的 长度 以及 存在时间 这是由于它必须生成相应的代码 以便向上和向下移动指针 这一限制无疑影响了程序的灵活性 所以尽管有些 Java数据要保存在堆栈里 特别是对象引用 但 Java 对象并不放到其中

  在函数中定义的一些基本类型的变量和对象的引用变量都在函数的堆栈 中分配 。

当在一段代码块定义一个变量时,Java就在栈中为这个变量分配内存空间,当超过变量的作用域后,Java会自动释放掉为该变量所分配的内存空间, 该内存空间可以立即被另作他用。 所以尽量使用基本类型的变量.

 

堆(或 内存堆 heap) 一种常规用途的内存池 也在 RAM 内 所有 Java 对象都保存在里面 和堆栈不同 内存堆 或 堆 Heap 最吸引人的地方在于编译器不必知道要从堆里分配多少存储空间 也不必知道存储的数据要在堆里呆多长的时间 因此 用堆保存数据时会得到更大的灵活性 要创建一个对象时 只需用 new 命令编制相关的代码即可执行这些代码时 就会在堆里自动进行数据的保存 不过 为了获得这种灵活性 我们也必然需要付出一定的代价 假如在内存堆里分配存储空间 和分配规格存储空间相比 前者要花掉更长的时间 和 C++不同 Java 事实上是不允许在堆栈里创建对象的 这样说 只是为了进行理论上的一种比较

堆内存用来存放由 new创建的对象和数组。 由Java虚拟机的自动垃圾回收器来管理。

   在堆中产生了一个数组或对象后,还可以在栈中定义一个特殊的变量,让栈中这个变量的取值等于数组或对象在堆内存中的首地址,栈中的这个变量就成了数组或对象的引用变量
   引用变量就相当于是为数组或对象起的一个名称,以后就可以在程序中使用栈中的引用变量来访问堆中的数组或对象

 

 

静态存储static storage  静态 Static 是指 位于固定位置 尽管仍在 RAM 里程序运行期间 静态存储的数据将随时等候调用 可用 static 关键字指出一个对象的特定元素是静态的 但 Java 对象本身永远都不会不会置入静态存储空间 

 

常数存储constant storage 常数值通常直接置于程序代码内部 这样做是安全的 因为它们永远都不会改变 有的常数需要严格地保护 所以可考虑将它们置入只读存储器 ROM

 

非 RAM 存储 若数据完全独立于一个程序之外 那么即使程序不运行了 它们仍可存在 并处在程序的控制范围之外 其中两个最主要的例子便是 流式对象  持久性对象 对于流式对象 对象会变成字节流 通常会发给另一台机器 而对于持久性对象我们可把它们保存在磁盘或磁带中 即使程序中止运行 它们仍可保持自己的状态不变 之所以要设计这些类型的数据存储 最主要的一个考虑便是把对象变成可在其他媒体上存在的形式 以后一旦需要 还可重新变回一个普通的 存在于 RAM 里的对象 目前 Java 只提供了有限的 持久性对象 支持 在未来的 Java 版本中 有望提供对 持久性 更完善的支持 。

 

转载别人总结的: 红色的为自己补充 (http://blog.csdn.net/gaowenming/archive/2010/02/22/5316423.aspx)

栈与堆都是Java用来在Ram中存放数据的地方。与C++不同,Java自动管理栈和堆,程序员不能直接地设置栈或堆。
      Java的堆或者说内存堆是一个运行时数据区,类的(对象从中分配空间。这些对象通过new、newarray、anewarray和multianewarray等 指令建立,它们不需要程序代码来显式的释放。堆是由垃圾回收来负责的,堆的优势是可以动态地分配内存大小,生存期也不必事先告诉编译器,因为它是在运行时 动态分配内存的,Java的垃圾收集器会自动收走这些不再使用的数据。但缺点是,由于要在运行时动态分配内存,存取速度较慢。
      堆栈的优势是,存取速度比堆要快,仅次于寄存器,栈数据可以共享。但缺点是,存在栈中的数据大小与生存期必须是确定的,缺乏灵活性。栈中主要存放一些基本类 型的变量(,int, short, long, byte, float, double, boolean, char)和对象句柄。
栈有一个很重要的特殊性,就是存在栈中的数据可以共享 。
 假设我们同时定义:
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, 它是由编译器完成的,它有利于节省空间。而一个对象引用变量修改了这个对象的内部状态,会影响到另一个对象引用变量。
String是一 个特殊的包装类数据。可以用:
String str = new String("abc");
String str = "abc";
两种的形式来创建,第一种是用new()来新 建对象的,它会在存放于堆中。每调用一次就会创建一个新的对象。

-> String str = new String("abc");自己补充: 应该说有会产生两个对象,一个为new String("abc")的实体对象放到内存堆中, 一个为堆栈对象 str 也就是类实例对象的引用对象。


而第二种String str = "abc";是先在栈中创建一个对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的方式是生成不同的对象。每一次生成一个 
   因此用第二种方式String str = "abc";创建多个”abc”字符串,在内存中其实只存在一个对象而已. 这种写法有利与节省内存空间. 同时它可以在一定程度上提高程序的运行速度,因为JVM会自动根据栈中数据的实际情况来决定是否有必要创建新对象。而对于String str = new String("abc");的代码,则一概在堆中创建新对象,而不管其字符串值是否相等,是否有必要创建新对象,从而加重了程序的负担。
   另一方面, 要注意: 我们在使用诸如String str = "abc";的格式定义类时,总是想当然地认为,创建了String类的对象str。担心陷阱!对象可能并没有被创建!而可能只是指向一个先前已经创建的 对象。只有通过new()方法才能保证每次都创建一个新的对象。由于String类的immutable性质,当String变量需要经常变换其值时,应 该考虑使用StringBuffer类,以提高程序效率。

Primitive类型 --》 放到堆栈中,可以参考上面的说明。(比较奇怪BigDecimal与date 类型怎么没在下面表中,另外 Stirng = “abc” 不是也放到堆栈中,所以String 是不是也可以说?)

 



 

 

  • 大小: 28.7 KB
  • 大小: 47.8 KB
分享到:
评论

相关推荐

    java实现内存动态分配

    Java 实现内存动态分配主要涉及Java内存模型以及内存管理机制,包括堆内存和栈内存的分配,以及垃圾回收等概念。下面将详细解释这些知识点。 1. **Java内存模型** Java程序运行时,内存分为堆内存(Heap)和栈内存...

    Java中堆内存与栈内存分配浅析

    ### Java中堆内存与栈内存分配浅析 #### 一、引言 在Java编程语言中,内存管理是一项至关重要的技术。程序运行时所使用的内存主要分为两类:堆内存(Heap Memory)和栈内存(Stack Memory)。理解这两种内存类型的...

    java实现的内存分配

    总的来说,理解和掌握Java中的内存分配策略对于编写高效、健壮的代码至关重要。无论是轮转法还是高优先权法,都有其适用的场景,开发者可以根据具体需求选择合适的策略,并利用Java提供的工具和机制进行优化。在实际...

    java中内存分配

    Java 中的内存分配是 Java 程序员必须掌握的重要知识。JAVA 中内存分配的问题是指 Java 程序在运行时如何在内存中存储数据的过程。Java 程序运行时有 6 个地方可以存储数据,它们分别是寄存器、栈、堆、静态存储、...

    深入Java核心_Java内存分配原理精讲

    此外,Java内存模型(JMM)也非常重要,它定义了线程之间的共享变量如何交互,以及在并发环境下如何保证内存可见性和一致性。熟悉 volatile、synchronized 和 final 关键字的作用,可以帮助我们编写出正确处理并发的...

    Java运行时内存分配

    ### Java运行时内存分配详解 #### 一、引言 Java作为一种广泛使用的编程语言,其内存管理机制一直是开发者关注的重点之一。Java程序在运行时如何分配内存?对象是如何存储的?这些问题不仅对于深入理解Java虚拟机...

    java内存分配情况

    Java内存分配主要涉及五个区域:寄存器、栈、堆、静态域和常量池。在Java编程中,理解这些内存区域的分配规则对于优化代码性能和避免内存泄漏至关重要。 1. **寄存器**:这是最快捷的存储区域,但不在Java程序员的...

    java程序中的内存分配问题

    Java程序在运行时的内存分配是一个关键的概念,它关乎...理解并掌握这些Java内存分配的知识点,对于编写高效、稳定的Java程序至关重要。通过实践和深入学习,开发者可以更好地控制程序的内存使用,减少错误,提升性能。

    java中jvm内存分配相关资料总结整理

    本资料总结主要关注JVM内存分配及其运行原理,这对于理解和优化Java应用程序的性能至关重要。 1. **JVM内存结构** JVM内存分为几个关键区域:方法区(Method Area)、堆(Heap)、栈(Stack)、程序计数器(PC ...

    java核心 内存分配问题

    Java 内存分配是理解Java程序性能和...理解Java内存分配对于编写高效、无内存泄漏的代码至关重要,同时也对诊断和解决性能问题有着深远的影响。通过深入了解这些概念,开发者能够更好地设计和优化他们的Java应用程序。

    Java内存分配浅析

    Java内存分配是Java编程中非常重要的概念,它涉及到程序运行时的数据存储和管理。Java程序在JVM(Java Virtual Machine,Java虚拟机)上运行,JVM作为一个平台无关的执行环境,负责Java程序的内存管理和执行。理解...

    java内存分配机制详解

    ### Java内存分配机制详解 #### 一、引言 Java作为一种广泛应用的编程语言,其内存管理机制对于确保程序高效稳定运行至关重要。本文旨在详细介绍Java内存分配机制中的几个关键概念:寄存器、栈、堆、静态域、常量...

    java内存分配详解

    ### Java内存分配详解 #### 一、Java内存区域划分 Java程序在运行过程中涉及的内存主要分为两大类:栈内存和堆内存。 1. **栈内存**:主要用于存储局部变量,如基本数据类型变量(int、long、char等)以及对象的...

    附录:Java的内存分配

    ### Java的内存分配详解 #### 一、Java内存模型概览 Java的内存管理是Java程序性能的关键之一。为了确保程序高效稳定地运行,开发者必须理解Java如何管理和分配内存资源。Java程序的内存主要分为以下几个部分: -...

    java内存对象分配过程研究

    深入理解Java中对象的内存分配过程对于编写高效稳定的程序至关重要。本文通过对Java虚拟机内存区域的分析,详细介绍了对象创建的过程,包括定义对象变量、为对象分配内存以及初始化对象等关键步骤。此外,还探讨了...

    深入Java核心 Java内存分配原理精讲

    Java内存分配机制是Java编程语言的重要组成部分,它不仅直接影响着程序的性能,也是理解Java垃圾回收、内存泄漏等问题的关键。本文将聚焦于Java内存分配的核心概念,包括栈、堆、常量池等关键区域的功能与使用场景,...

    Java内存分配原理精讲

    ### Java内存分配原理精讲 #### 一、引言 Java作为一门广泛应用于企业级开发的语言,其内存管理和分配机制是其核心技术之一。本文旨在深入探讨Java内存分配的基本原理及其在不同内存区域的具体表现,帮助读者更好...

    关于Java变量的声明、内存分配及初始化详解

    本文将深入探讨Java变量的声明、内存分配以及初始化。 首先,变量的声明是指在程序中定义一个变量,指定其类型和名称。例如,`String name;` 和 `int age;` 在`Person`类中声明了两个变量,`name`是一个字符串类型...

Global site tag (gtag.js) - Google Analytics