一、内存(1-4为运行时数据区)
1. 程序计数器
一块较小的内存空间,是当前线程所执行的字节码的行号指示器。
2. 栈
虚拟机栈:为虚拟机执行java方法服务
本地方法栈:为虚拟机中使用到的native方法服务
它们都是线程私有的。
3. 堆
被所有线程共享(也可能分配出多个线程私有的分配缓冲区)
所有对象实例以及数组都在堆上分配(目前开始不这么绝对了(各种优化技术))
4. 方法区(别名non-heap)
跟堆一样,也是线程共享的。存储已被加载的类信息、常量、静态变量、即时编译器编译后的代码
此区域的垃圾回收主要针对常量池的回收和对类型的卸载。
5. 运行时常量池
是方法区的一部分
存放编译期生成的各种字面量和符号引用,会在类加载时进入方法区的运行时常量池中存放。
String的方法intern()——扩充常量池:会查看常量池中是否存在与对象相同Unicode的字符串常量,如果有,则返回其引用;如果没有,则在常量池中新增一个Unicode等于对象字符串并返回它的引用,但这个之前不在常量池中的对象仍然存在。
6. 直接内存
不是虚拟机运行时数据区的一部分,也不是java虚拟机规范中定义的内存区域,它的分配不受java堆大小的限制。
二、内存管理
分配方式:“指针碰撞”(堆中内存规整)、“空闲列表”(堆中内存不规整)。选择哪种方式由垃圾收集器是否有整理。例如serial、parnew等带compact过程的是有的,而CMS这种基于Mark-Sweep算法的是没有的。
并发情况下,对象创建,可能会导致内存指针紊乱。解决此问题:1、对分配内存空间的动作进行同步处理;2、设置本地线程分配缓冲,用来单独分配内存。
创建对象,执行new指令之后会接着执行<init>,这样真正可用的对象才算完全产生出来。
三、内存溢出与内存泄漏的区别
内存溢出 out of memory,是指程序在申请内存时,没有足够的内存空间供其使用,出现out of memory;比如申请了一个integer,但给它存了long才能存下的数,那就是内存溢出。
内存泄露 memory leak,是指程序在申请内存后,无法释放已申请的内存空间,一次内存泄露危害可以忽略,但内存泄露堆积后果很严重,无论多少内存,迟早会被占光。
memory leak会最终会导致out ofmemory!
内存溢出就是你要求分配的内存超出了系统能给你的,系统不能满足需求,于是产生溢出。
内存泄漏是指你向系统申请分配内存进行使用(new),可是使用完了以后却不归还(delete),结果你申请到的那块内存你自己也不能再访问(也许你把它的地址给弄丢了),而系统也不能再次将它分配给需要的程序。一个盘子用尽各种方法只能装4个果子,你装了5个,结果掉倒地上不能吃了。这就是溢出!比方说栈,栈满时再做进栈必定产生空间溢出,叫上溢,栈空时再做退栈也产生空间溢出,称为下溢。就是分配的内存不足以放下数据项序列,称为内存溢出.