文章目录

  • 运行时数据区域:
  • 内存溢出和内存泄漏:


运行时数据区域:

1、程序计数器:
		线程私有。
		为了线程切换后,恢复到正确的执行位置。
		不会出现OutOfMemoryError。
2、虚拟机栈:
		线程私有。
		生命周期与线程相同。
		服务于字节码文件。(.class文件)
		会出现两种情况:
			栈内存溢出StackOverFlowError:
				发生于方法的无限递归调用。
			内存溢出OutOfMemoryError:
				程序在申请内存后,无法释放已申请的内存。
3、本地方法栈:
		服务于本地方法。
		线程私有。
		会出现两种情况:
			栈内存溢出StackOverFlowError:
				发生于方法的无限递归调用。
			内存溢出OutOfMemoryError:
				程序在申请内存后,无法释放已申请的内存。
4、堆:
		内存最大。
		线程共享。
		垃圾收集器管理的内存区域,又称GC堆。
		虚拟机启动时创建。
		存放对象实例。
		分代收集:
			新生代、老年代、永久代...
		分代收集的好处:
			为了更好地回收内存/更快分配内存。
		通过参数扩展大小:
			-Xmx、-Xms
		物理上不连续的内存空间。
5、方法区:
		线程共享。
		存放:
			类型信息、常量、静态变量、编译后的代码缓存等
		JDK8以前:
			方法区又称为永久代。
			但是这样会造成内存溢出。(永久代有-XX:MaxPermSize的上限)
		JDK8:
			放弃了永久代。改为元空间代替。
		JDK6的时候,有放弃永久代,采用本地内存来实现方法区的计划。
		JDK7的时候,将字符串常量池、静态变量移至java堆中。将类型信息移至元空间。
6、常量池:
		是方法区的一部分。
		用于存放编译期生成的各种字面量和符号引用。
		常量池无法再申 请到内存时会抛出OutOfMemoryError异常。
		具备动态性;运行期间也可能将新的常量放入池中;用得比较多的便是String类的intern()方法。
7、直接内存:
		不属于运行时数据区的一部分。
		会频繁使用,会造成OutOfMemoryError。

内存溢出和内存泄漏:

内存溢出OutOfMemoryError:
	程序在申请内存时,没有足够的内存空间供其使用。
内存泄漏MemoryLeak:
	程序在申请内存后,无法释放已申请的内存。
内存泄漏会最终导致内存溢出。