你好,游客 登录 注册 搜索
背景:
阅读新闻

JVM 运行时数据区域

[日期:2017-08-14] 来源:Linux社区  作者:fhhk [字体: ]

程序计数器(Program Counter Register)

        像我们平时读书一样,当我们在去做别的事情之前,我们会对我们读到什么地方了做一个标记,方便我们再回来的时候接着重新读。如果这本书有很多人读呢?那么每个人都会对自己读到了哪里做一个标记。(标记都是个人自己保存)

        那么程序计数器也是一样的,同样的代码,可能会被多个线程执行,那么每个线程都要记住自己执行到哪行代码了。这样在线程再次切换执行的时候,知道代码接着从哪里执行。

        所以程序计数器是线程私有的,每个线程都会程序计数器。对于正在执行的Native方法,这个计数器值为空。

    虚拟机栈(VM Stack)

        与程序计数器一样,Java虚拟机栈也是线程私有的。每个线程执行的时候都会创建一个栈帧(Stack Frame)。用来存放局部变量表、操作数栈、动态链接、方法出口等信息。每个方法的调用到执行完成的过程,都对应着一个栈帧在虚拟机中入栈到出栈的过程。

  局部变量表

        局部变量表中存放了编译器可知的各种数据类型,和对象引用类型。其中除了64位长度的long和double类型会占用2个局部变量空间(Slot),其余的数据类型只占用1个。所以每个方法需要分配多大的局部变量表空间是完全确定的,在编译期间就完成分配。在方法运行期间不会发生改变。

本地方法栈(Native Method Stack)

        与虚拟机栈一样,只是虚拟机栈是为虚拟机执行的Java方法服务,而本地方法栈为虚拟机中使用到的Native方法服务。

       Java堆,是被所有线程共享的区域,在虚拟机启动时创建。用来存放对象示例。

  随着对象的创建,Java堆中的对象实例会越来越多,会造成内存溢出。在这些对象中,有百分之八十都是使用完之后就不用了,所以我们可以对这些对象进行回收。就引出了垃圾回收机制,对堆中没用的对象进行清理。为了方便对象清理,又把堆分为了新生代和老年代。其中新生代又可以详细分为Eden区、From Survive区和To Survive区。

方法区 

        在方法区存储了类的信息,静态变量、常量等数据。虽然Java虚拟机规范把方法区描述为堆的一个逻辑部分,但是它有一个别名Non-Heap(非堆),目的是为了和Java堆区分开。

        在HotSpot虚拟机上,方法区也被称为“永久代”,这与Java虚拟机的垃圾回收有关。但是这样设计并不好,更加容易导致内存溢出问题。HotSpot虚拟机现在也有放弃永久代并逐步改为采用Native Memory来实现方法区的规划。从JDK1.7的HotSpot中,已经把原本在永久代的字符串常量池移出。

 运行时常量池

        运行时常量池也是方法区的一部分。用来存放编译器生成的各种字面量和符号引用。Java语言并不要求常量池一定只有编译器才能产生,运行期间也可能将新的常量放入池中。利用的最多的就是String的intern()方法。

直接内存

        直接内存并不是虚拟机运行时数据区的一部分,也不是Java虚拟机规范中定义的内存区域。但是这部分内存被频繁使用,也可能导致内存溢出异常出现。     

本文永久更新链接地址http://www.linuxidc.com/Linux/2017-08/146218.htm

linux
相关资讯       JVM  JVM 运行时数据区域 
本文评论   查看全部评论 (0)
表情: 表情 姓名: 字数

       

评论声明
  • 尊重网上道德,遵守中华人民共和国的各项有关法律法规
  • 承担一切因您的行为而直接或间接导致的民事或刑事法律责任
  • 本站管理人员有权保留或删除其管辖留言中的任意内容
  • 本站有权在网站内转载或引用您的评论
  • 参与本评论即表明您已经阅读并接受上述条款