java内存模型,这里其实是指 jvm运行时内存模型
1 每一个应用程序,都有一个JVM
而不是 多个应用程序,共享一个jvm
2 java源文件,首先通过编译器,把java语法的代码,编译成 jvm语法的字节码文件
这个过程,是不涉及到jvm的。???
然后,jvm通过类加载,把需要的类字节码文件,加载进内存中。
3 jvm运行时内存分为两部分:线程共享内存和线程私有内存
线程共享内存包括:堆、方法区(包含 运行时常量池)
线程非共享内存包括:java栈,本地方法栈,PC程序寄存器
~~~~~~~~~~~~~
每个线程,都有一份独有的线程非共享内存。
pc程序寄存器,记录某个线程当前执行到哪条字节码
本地方法栈类似于java栈,但是是为了native方法准备的
java栈,很重要。每个线程都有一个java栈,它是为java方法准备的。栈里面存储着一个个 栈帧,每一个栈帧可看做一个方法的调用
栈帧包括:局部变量表,操作数栈,方法返回地址,其他信息 。其中操作数栈,才是和cpu中的ALU联系在一起的。是cpu唯一指定的数据来源。 执行一条指令,其中涉及到的操作,就是在 (局部变量表和操作数栈中进行的)
~~~~~~~~~~~~~
方法区,不是执行方法的。执行方法的内存,是在 java栈中。
方法区是涉及到类加载的时候,加载进来的类的信息,常量,字段,方法代码等的信息
而运行时常量池是方法区的一部分。其中存储的是,字面常量,符号引用等?、、
符号引用:是字符串,其中有丰富的信息,通过搜索类的方法表,可以用来定位到方法。
符号引用只需要第一次用后,就会被替换成 直接引用了。
当加载的类很多的时候,方法去也会溢出。
~~~~~~~~~~~~~~~~~~~~~~~~~~~~
线程私有内存:java栈,本地方法栈,这些都和线程有关,和线程的生命周期相同,一旦线程结束,相应的内存也将释放,所以这部分内存不需要关心回收
我们的垃圾回收主要是针对 堆内存。不过,方法区的内存是永久区,一般也不需要考虑回收???
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
参考的网站和帖子
:https://www.zhihu.com/question/19748817
https://www.zhihu.com/question/21539353
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
参考知乎回复
1 类加载后,类信息放在方法区,也就是所谓的PermGen的,如果真像题主说的无限类,必然出现OOM
2 其他参考