线程私有的数据区域的生命周期与线程相同,依赖用户线程的 ‘启动/结束’ 而 ‘创建/销毁’,(在Hotspot VM内,每个线程都与操作系统的本地线程直接映射,因此部分线程私有 的内存区域随着线程的’创建/销毁’而’创建/销毁’)
线程私有
1.程序计数器
一块较小的内存空间,可以看作当前线程所执行的字节码行号的指示器(想象debug时的场景),若执行的是Native方法,则计数器的值为空,这也是JAVA虚拟机规范中唯一一个没有规定任何OutOfMemoryError情况的区域。
由于线程是轮流切换并分配处器执行时间的方式来实现的,每个时间点,一个处理器都只会执行一个线程的指令,为了保证每次线程切换,指令的正确执行,故程序计数器为线程私有。
2.虚拟机栈(VM Stack)
Java方法的内存模型,每个方法在执行的同时都会创建一个栈帧(Stack Frame),用于存储局部变量表、操作数栈、动态链接、方法出口等信息。每一个方法从调用到执行完成,就对应着一个栈帧在虚拟机栈中入栈到出栈的过程。
局部变量表存放了编译期可知的各种基础数据类型、对象引用(指向对象的起始地址的指针,或一个代表对象的句柄,或其它与此对象相关的位置)
3.本地方法栈(Native Method Stack)
对应虚拟机栈,虚拟机栈存储的是JAVA方法的相关信息,本地方法栈存放的是Native方法的相关信息,本地方法栈在虚拟机规范中只定义了一个规范,具体实现由各个虚拟机厂商实现的。
线程共享
4.堆(Java Heap)
Java虚拟机堆所管理的是内存中最大的一块,主要用于存放对象实例,几乎所有的对象实例都要在堆上分配,因此也是GC(垃圾回收机制)工作的主要区域。
Java堆从GC的角度看,可以分为新生代和老年代,再细致一点可以分为Eden空间、From Survivor 空间、To Survivor空间等。从内存分配的角度来看,线程共享的Java堆中可能划分出多个线程私有的分配缓冲区。
5.方法区(Method Area)
用于存放已被JVN加载的类信息、常量、静态变量等。
6.运行时常量池
方法区的一部分,用于存放编译时产生的各种字面量和符号引用。
7.直接内存
直接内存并不是JVM运行时数据区的一部分,但也会被频繁的使用。在JDK1.4中引入的NIO,提供了基于Channel与Buffer的IO方式,他可以使用Native方法直接分配堆外的内存,然后使用DirectByteBuffer对象作为这块内存的引用进行操作,这样避免了JAVA堆和Native方法堆中来回复制数据,可以显著提高性能。
NIO和IO区别:
1.IO面向流,NIO面向缓冲区;
2.IO是阻塞的,NIO是非阻塞的;
3.NIO允许一个单独的线程来监视多个输入通道;