JVM 内存初学 (堆(heap)、栈(stack)和方法区(method) )
1.堆:存储对象(实例),被所有线程共享
2.栈:存储对象引用和基本数据类型,每个线程都有一个线程栈
3.方法区:也叫静态区,存储class信息和static变量,被所有线程共享
public class Test {
public static void main(String[] args) {
public Test2 t2 = new Test2();
//JVM将Test2类信息加载到方法区,new Test2()实例保存在堆区,Test2引用保存在栈区
}
}
Original Address:http://chinagdvea.iteye.com/blog/829766
******************************************************************************************************************************************
这两天看了一下深入浅出JVM这本书,推荐给高级的java程序员去看,对你了解JAVA的底层和运行机制有
比较大的帮助。
废话不想讲了.入主题:
先了解具体的概念:
JAVA的JVM的内存可分为3个区:堆(heap)、栈(stack)和方法区(method)
堆区:
1.存储的全部是对象,每个对象都包含一个与之对应的class的信息。(class的目的是得到操作指令)
2.jvm只有一个堆区(heap)被所有线程共享,堆中不存放基本类型和对象引用,只存放对象本身
栈区:
1.每个线程包含一个栈区,栈中只保存基础数据类型的对象和自定义对象的引用(不是对象),对象都存放在堆区中
2.每个栈中的数据(原始类型和对象引用)都是私有的,其他栈不能访问。
3.栈分为3个部分:基本类型变量区、执行环境上下文、操作指令区(存放操作指令)。
方法区:
1.又叫静态区,跟堆一样,被所有的线程共享。方法区包含所有的class和static变量。
2.方法区中包含的都是在整个程序中永远唯一的元素,如class,static变量。
为了更清楚地搞明白发生在运行时数据区里的黑幕,我们来准备2个小道具(2个非常简单的小程序)。
AppMain.java
public class AppMain //运行时, jvm 把appmain的信息都放入方法区
{
public static void main(String[] args) //main 方法本身放入方法区。
{
Sample test1 = new Sample( " 测试1 " ); //test1是引用,所以放到栈区里, Sample是自定义对象应该放到堆里面
Sample test2 = new Sample( " 测试2 " );
test1.printName();
test2.printName();
}
}
Sample.java
public class Sample //运行时, jvm 把appmain的信息都放入方法区
{
/** 范例名称 */
private name; //new Sample实例后, name 引用放入栈区里, name 对象放入堆里
/** 构造方法 */
public Sample(String name)
{
this .name = name;
}
/** 输出 */
public void printName() //print方法本身放入 方法区里。
{
System.out.println(name);
}
}
OK,让我们开始行动吧,出发指令就是:“java AppMain”,包包里带好我们的行动向导图,Let’s GO!
系统收到了我们发出的指令,启动了一个Java虚拟机进程,这个进程首先从classpath中找到AppMain.class文件,读取这个文件中的二进制数据,然后把Appmain类的类信息存放到运行时数据区的方法区中。这一过程称为AppMain类的加载过程。
接着,Java虚拟机定位到方法区中AppMain类的Main()方法的字节码,开始执行它的指令。这个main()方法的第一条语句就是:
Sample test1=new Sample("测试1");
语句很简单啦,就是让java虚拟机创建一个Sample实例,并且呢,使引用变量test1引用这个实例。貌似小case一桩哦,就让我们来跟踪一下Java虚拟机,看看它究竟是怎么来执行这个任务的:
1、 Java虚拟机一看,不就是建立一个Sample实例吗,简单,于是就直奔方法区而去,先找到Sample类的类型信息再说。结果呢,嘿嘿,没找到@@,这会儿的方法区里还没有Sample类呢。可Java虚拟机也不是一根筋的笨蛋,于是,它发扬“自己动手,丰衣足食”的作风,立马加载了Sample类,把Sample类的类型信息存放在方法区里。
2、 好啦,资料找到了,下面就开始干活啦。Java虚拟机做的第一件事情就是在堆区中为一个新的Sample实例分配内存, 这个Sample实例持有着指向方法区的Sample类的类型信息的引用。这里所说的引用,实际上指的是Sample类的类型信息在方法区中的内存地址,其实,就是有点类似于C语言里的指针啦~~,而这个地址呢,就存放了在Sample实例的数据区里。
3、 在JAVA虚拟机进程中,每个线程都会拥有一个方法调用栈,用来跟踪线程运行中一系列的方法调用过程,栈中的每一个元素就被称为栈帧,每当线程调用一个方法的时候就会向方法栈压入一个新帧。这里的帧用来存储方法的参数、局部变量和运算过程中的临时数据。OK,原理讲完了,就让我们来继续我们的跟踪行动!位于“=”前的Test1是一个在main()方法中定义的变量,可见,它是一个局部变量,因此,它被会添加到了执行main()方法的主线程的JAVA方法调用栈中。而“=”将把这个test1变量指向堆区中的Sample实例,也就是说,它持有指向Sample实例的引用。
OK,到这里为止呢,JAVA虚拟机就完成了这个简单语句的执行任务。参考我们的行动向导图,我们终于初步摸清了JAVA虚拟机的一点点底细了,COOL!
接下来,JAVA虚拟机将继续执行后续指令,在堆区里继续创建另一个Sample实例,然后依次执行它们的printName()方法。当JAVA虚拟机执行test1.printName()方法时,JAVA虚拟机根据局部变量test1持有的引用,定位到堆区中的Sample实例,再根据Sample实例持有的引用,定位到方法区中Sample类的类型信息,从而获得printName()方法的字节码,接着执行printName()方法包含的指令。
Original Address:http://www.blogjava.net/mlzry0612/articles/223420.html
- 大小: 34.5 KB
分享到:
相关推荐
本手册旨在为初学者提供关于 JVM 的基础知识,重点介绍 JVM 内存模型中的关键概念。 #### JVM 内存模型概览 JVM 在执行 Java 程序的过程中,将其所管理的内存划分为几个不同的区域,每个区域有其特定的功能和生命...
2. **运行时数据区(Runtime Data Area)**:包括堆(Heap)、方法区(Method Area)、虚拟机栈(JVM Stack)、本地方法栈(Native Method Stack)和程序计数器(PC Register)。 3. **执行引擎(Execution Engine)...
2. **运行时数据区(Run-Time Data Areas)**:包括方法区、堆、虚拟机栈、本地方法栈和程序计数器。 3. **执行引擎(Execution Engine)**:执行字节码,包括解释器和即时编译器(JIT)。 4. **本地方法接口(Native ...
在运行过程中,JVM会管理内存,包括堆(Heap)、栈(Stack)以及方法区(Method Area),这些区域各有其特定的功能和限制。 堆栈溢出(Stack Overflow)是程序运行时常见的错误,通常发生在当一个函数或方法的递归...
- **3.2.3 本地方法栈(Native Method Stack)**:与虚拟机栈相似,但它服务于Native方法。同样,本地方法栈也会抛出相同的异常。 - **3.2.4 堆(Heap)**:是JVM管理的最大内存区域,所有的对象实例都存储在这里。堆被...
4. **本地方法栈(Native Method Stack)**:支持本地方法(用C/C++实现的)的执行,与Java栈类似,但服务于本地方法。 在多线程环境下,Java内存模型确保了线程之间的通信和同步。比如,通过监视器锁(Monitor ...
JVM的内存模型包括堆(Heap)、栈(Stack)、方法区(Method Area)、程序计数器(PC Register)和本地方法栈(Native Method Stack)。其中,堆是所有对象的存储空间,而栈则存储方法调用的状态。 垃圾回收(GC)...
在Java内存模型中,存在三个重要的区域:程序计数器(PC Register)、虚拟机栈(JVM Stack)和本地方法栈(Native Method Stack),这些都是线程私有的;还有堆(Heap)和方法区(Method Area),是所有线程共享的。...
5. 本地方法栈(Native Method Stack):与JVM栈类似,但用于支持本地方法(用C++或其他语言实现的Java库)的执行。 三、垃圾回收(Garbage Collection) JVM自动管理内存,通过垃圾回收机制回收不再使用的对象所...
JVM采用栈式架构,其中虚拟机栈和本地方法栈用于存储线程的局部变量和操作数。每当执行一个方法时,JVM会创建一个新的栈帧,当方法执行完毕,该栈帧会被销毁。 #### 内存管理与垃圾回收 JVM的内存管理主要包括对象...
JVM的内存区域主要包括堆内存(Heap)、方法区(Method Area)、栈(Stack)、本地方法栈(Native Method Stack)以及程序计数器(Program Counter Register)。每种内存区域都有其特定的作用和管理方式。 - **堆内存(Heap)**...
- 本地方法栈(Native Method Stack):与虚拟机栈类似,但主要用于存储调用本地方法的信息。 - 程序计数器(Program Counter Register):记录当前线程所执行的字节码指令的位置。 3. **执行引擎(Execution ...
JVM内存主要分为堆内存(Heap)、栈内存(Stack)、方法区(Method Area)、程序计数器(PC Register)和本地方法栈(Native Method Stack)。其中,对象实例主要在堆内存中分配,而方法调用时的局部变量存储在栈中...
3. 内存管理(Memory Management):负责管理运行时数据区,包括堆(Heap)、方法区(Method Area)、虚拟机栈(VM Stack)、本地方法栈(Native Method Stack)、程序计数器(Program Counter)。 4. 本地接口...
1. **内存区域**:Java程序运行时涉及堆内存(Heap)、栈内存(Stack)、方法区(Method Area)和本地方法栈(Native Method Stack)等不同区域。源代码可能展示了如何在这些区域中分配和管理内存。 2. **对象生命...
4. **内存模型**:JVM有多个内存区域,包括堆(Heap)、栈(Stack)、方法区(Method Area)、程序计数器(PC Register)和本地方法栈(Native Method Stack)。每个区域都有其特定的作用和生命周期。 5. **垃圾...
- **方法区(method area)**:也被称作“非堆”,它与堆一样,会在JVM启动时创建。它用于存储已被虚拟机加载的类信息、常量、静态变量、即时编译后的代码等数据。 - **程序计数器(Program Counter Register)**:当前...
它主要包括堆(Heap)、栈(Stack)、方法区(Method Area)、程序计数器(PC Register)、本地方法栈(Native Method Stack)等几个部分。堆用于存储所有对象实例,栈则存储方法调用时的局部变量,方法区存储类的...
- **栈(stack)**:主要用于存储基本数据类型和对象引用,每条线程拥有一个独立的栈空间。 - **方法区(Method Area)**:存储已加载类的信息、常量、静态变量等数据。 #### 七、垃圾回收机制 - **垃圾回收(Garbage ...
- **栈(Stack)**:用于存储线程的局部变量和方法调用栈。 - **方法区(Method Area)**:存放类信息、常量、静态变量等数据。 - **程序计数器(Program Counter Register)**:指向当前线程正在执行的字节码指令。...