我们定义一个基本数据类型的变量,一个对象的引用,函数调用的现场都保存使用JVM中的栈内存
通过new关键字和构造器创建的对象则放在堆内存,堆是垃圾收集器管理的主要区域,由于现在的垃圾收集器都采用分代收集算法,堆内存还可以细分为新生代和老生代
再具体一点可以分为Eden、Survior、又可分为From Survior和To Survior、Teneurd
方法区和堆都是各个线程共享的内存区域,用于存储已经被JVM加载的类信息,常量,静态变量,JIT编译器编译后的代码等数据,常量池也是方法区的一部分
栈内存操作起来最快但是栈很小,一般大量的对象都是放在堆内存,栈和堆的内存都可以通过JVM的启动参数来调整
栈用光了会引发StackOverflowError,而堆和常量池不足则会引发OutOfMemoryError
较新版本的Java中由于JIT编译器的发展和“逃逸分析”技术逐渐成熟,栈上分配、标量替换等优化技术使得对象一定分配在堆上这件事已经不是很绝对了
运行时常量池相当于Class文件常量池具有动态性,Java语言并不要求常量一定只有编译期间才能产生,运行期间也可以将新的常量放入池中,String类型的intern方法就是这样的