java内存模型和Jvm内存分配

java 内存模型(JMM): 

    线程之间的共享变量存储在主内存(main memory)中,每个线程都有一个私有的本地内存(local memory),本地内存中存储了该线程以读/写共享变量的副本。同过操作该副本,实现线程之间变量值的传递。    

(百度上说:在java程序运行时,jvm会在系统内存中划定一块区域作为程序的主内存, java中的所有变量都存在主内存中,并且对所有线程共享。而每个线程又有自己 的工作内存,工作内存中保存的是主内存中某些变量的拷贝,线程对多所有变量的操作都是发生在工作内存中的,线程之间不能直接访问,变量在程序中的传递依赖主内存。)

Jvm内存分配:

根据java 1.5 关于jvm 的规定 :

        jvm 将java程序运行时的内存分为如才几个区进行管理:

《java内存模型和Jvm内存分配》

栈区: 
栈分为java虚拟机栈和本地方法栈

  • 重点是Java虚拟机栈,它是线程私有的,生命周期与线程相同。
  • 每个方法执行都会创建一个栈帧,用于存放局部变量表,操作栈,动态链接,方法出口等。每个方法从被调用,直到被执行完。对应着一个栈帧在虚拟机中从入栈到出栈的过程。
  • 通常说的栈就是指局部变量表部分,存放编译期间可知的8种基本数据类型,及对象引用和指令地址。局部变量表是在编译期间完成分配,当进入一个方法时,这个栈中的局部变量分配内存大小是确定的。
  • 会有两种异常StackOverFlowError和 OutOfMemoneyError。当线程请求栈深度大于虚拟机所允许的深度就会抛出StackOverFlowError错误;虚拟机栈动态扩展,当扩展无法申请到足够的内存空间时候,抛出OutOfMemoneyError。
  • 本地方法栈 为虚拟机使用到本地方法服务(native)

堆区:

  • 堆被所有线程共享区域,在虚拟机启动时创建,唯一目的存放对象实例
  • 堆区是gc的主要区域,通常情况下分为两个区块年轻代和年老代。更细一点年轻代又分为Eden区最要放新创建对象,From survivor 和 To survivor 保存gc后幸存下的对象,默认情况下各自占比 8:1:1。 
    不过很多文章介绍分为3个区块,把方法区算着为永久代。这大概是基于Hotspot虚拟机划分, 然后比如IBM j9就不存在永久代概论。不管怎么分区,都是存放对象实例。
  • 会有异常OutOfMemoneyError

方法区:

  • 被所有线程共享区域,用于存放已被虚拟机加载的类信息,常量,静态变量等数据。被Java虚拟机描述为堆的一个逻辑部分。习惯是也叫它永久代(permanment generation)
  • 垃圾回收很少光顾这个区域,不过也是需要回收的,主要针对常量池回收,类型卸载。
  • 常量池用于存放编译期生成的各种字节码和符号引用,常量池具有一定的动态性,里面可以存放编译期生成的常量;运行期间的常量也可以添加进入常量池中,比如string的intern()方法。

程序计数器

  • 当前线程所执行的行号指示器。通过改变计数器的值来确定下一条指令,比如循环,分支,跳转,异常处理,线程恢复等都是依赖计数器来完成。
  • Java虚拟机多线程是通过线程轮流切换并分配处理器执行时间的方式实现的。为了线程切换能恢复到正确的位置,每条线程都需要一个独立的程序计数器,所以它是线程私有的。
  • 唯一一块Java虚拟机没有规定任何OutofMemoryError的区块
    原文作者:java内存模型
    原文地址: https://blog.csdn.net/imooc_000000/article/details/80295100
    本文转自网络文章,转载此文章仅为分享知识,如有侵权,请联系博主进行删除。
点赞