深入理解java虚拟机(一)—jvm内存模型总结

 本猿2013年毕业,到现在已经有五年了,虽然一直从事java工作,但是一直是写业务代码,设计业务产品那种,工作节奏比较散漫,最近才深深的感觉到我的能力跟毕业一两年的没多少差别,刚好最近工作闲了下来,开始看别人的一些博客,看了一些感悟,真的很受打击,不信你们看看,原来不知不觉我已经沦为不思进取,不求上进的人了

http://www.zuoxiaolong.com/html/article_184.html#

https://blog.csdn.net/chenssy/article/details/53738264


最近开始疯狂买书补救,看书如果不去总结,那么理论的知识肯定过不了几天就会忘掉,我打算重新开始写博客了,虽然我的文笔比较菜,总结能力也不强,不管了,自己写的自己能看懂就好了

 第一篇就写一写java虚拟机内存模型,简称JMM的结构吧:

JVM内存模型主要分为以下几个区域:

程序计数器,java虚拟机栈,本地方法栈,堆内存,方法区,运行时常量池


程序计数器

 程序计数器是线程私有的区域,很好理解嘛~,每个线程当然得有个计数器记录当前执行到那个指令。占用的内存空间小,可以把它看成是当前线程所执行的字节码的行号指示器。如果线程在执行Java方法,这个计数器记录的是正在执行的虚拟机字节码指令地址;如果执行的是Native方法,这个计数器的值为空(Undefined)。此内存区域是唯一一个在Java虚拟机规范中没有规定任何OutOfMemoryError情况的区域。(这段是抄别人的,原谅我自己表述会误导人)


虚拟机栈

虚拟机栈也是线程私有的,就是一个线程会分配一个栈,用于存放基本类型,引用类型的栈帧

  1. 局部变量表的创建是在方法被执行的时候,随着栈帧的创建而创建。而且,局部变量表的大小在编译时期就确定下来了,在创建的时候只需分配事先规定好的大小即可。此外,在方法运行的过程中局部变量表的大小是不会发生改变的。
  2. Java虚拟机栈会出现两种异常:StackOverFlowError和OutOfMemoryError。 
    a) StackOverFlowError: 
    若Java虚拟机栈的内存大小不允许动态扩展,那么当线程请求栈的深度超过当前Java虚拟机栈的最大深度的时候,就抛出StackOverFlowError异常。 
    b) OutOfMemoryError: 
    若Java虚拟机栈的内存大小允许动态扩展,且当线程请求栈时内存用完了,无法再动态扩展了,此时抛出OutOfMemoryError异常。
  3. Java虚拟机栈也是线程私有的,每个线程都有各自的Java虚拟机栈,而且随着线程的创建而创建,随着线程的死亡而死亡。

注:StackOverFlowError和OutOfMemoryError的异同? 
StackOverFlowError表示当前线程申请的栈超过了事先定好的栈的最大深度,但内存空间可能还有很多。 

而OutOfMemoryError是指当线程申请栈时发现栈已经满了,而且内存也全都用光了。


本地方法栈

本地方法栈与虚拟机栈 是比较类似的,都会抛出StackOverFlowError异常和OutOfMemoryError异常,本地方法栈是为本地方法Native创建的,在hotspot虚拟机中,本地方法栈与虚拟机栈是在同一区域,,,待补充

堆内存

这个区域就是jvm的仓库,基本上大部分的对象都存储在这个区域,而且这个区域是线程共享的,不像程序计数器或者虚拟机栈是线程唯一的,这个区域也是垃圾回收最活跃的场所,也最容易发生内存泄露,OOM的部分,一般虚拟机把这个区分去成 年轻代和老年代,年轻代一般是用完就回收的局部变量,老年代是逃过几次GC回收,多次被引用的对象存放的区域。堆内存不足的时候,就是无法分配内存去创建对象,就会抛出OutOfMemoryError: java heap space ,这个时候说明堆内存不足了,需要检查代码是不是有大量对象生成,或者存在内存泄露,

通过调节 -Xms,-Xmx来调整堆的大小,前提是要根据物理机的实际内存来调整。


方法区

方法区是存储类的class文件内容和数据结构,静态变量,常量,可以参考类加载器的加载过程。jvm加载class文件时,会把class文件的字节流内容和文件结构,转换成方法区的数据结构和内容,类加载时会生产一个class对象作为外部调用类方法的入口,这是唯一存储在方法区的对象吧。方法区有的地方又称 “永久代”。方法区内存不够用的时候也会报 java.lang.OutOfMemoryError: PermGen space 的错误。可以通过调节-XX:PermSize=64 M -XX:MaxPermSize=128 M 参数来调大方法区的内存。

运行时常量池

是存在方法区的一小块区域,一般是存放类编译的时候的常量或者符号引用,例如有些基本的数据就存储在这里

Byte,Short,Integer,Long,Character,Boolean 默认创建了数值在 -128-127区域的对应的缓存

Integer a = 127;

Integer b =127;

system.out.print(a==b)  ——> true

Integer a = 150;

Integer b =150;

system.out.print(a==b)  ——> false


直接内存

这个区域不在虚拟机的掌控范围,比如实现NIO时会直接使用直接内存,具体的查资料好吧。


好吧,第一篇就写到这里,感觉自己真的好水,还是科班出身的,如果你跟我类似,工作了四五年,还是混在小公司,享受着朝九晚五,又有些想法的话,不如跟我一起对java有一个深入的学习,不管技术能提高多少,至少我们努力过。

最近在看几本书,收货确实挺大的,推荐一下:

java 并发编程实战

深入理解java虚拟机

effectivjava

慢慢看吧,

不喜欢勿喷啊,这些大部分都是我靠脑子回忆的,有些细节还是翻书找的,今年开始java深入学习之路

下一篇继续讲jvm GC的内容






你可能感兴趣的:(深入理解java虚拟机)