GC机制:java垃圾回收机制,垃圾收集器线程(Garbage Collection Thread)在 JVM 处于空闲循环式,会自动回收无用的内存块。
垃圾收集算法:1、引用计数 2、根搜索 3、标记-清除 4、复制 5、标记-整理 6、分代收集
内存管理:JVM将内存划分为6个部分:PC寄存器(也叫程序计数器)、虚拟机栈、堆、方法区、运行时常量池、本地方法栈
JAVA的JVM的内存可分为3个区:堆(heap)、栈(stack)和方法区(method)
堆区:
1.存储的全部是对象,每个对象都包含一个与之对应的class的信息。(class的目的是得到操作指令)
2.jvm只有一个堆区(heap)被所有线程共享,堆中不存放基本类型和对象引用,只存放对象本身
栈区:
1.每个线程包含一个栈区,栈中只保存基础数据类型的对象和自定义对象的引用(不是对象),对象都存放在堆区中
2.每个栈中的数据(原始类型和对象引用)都是私有的,其他栈不能访问。
3.栈分为3个部分:基本类型变量区、执行环境上下文、操作指令区(存放操作指令)。
方法区:
1.又叫静态区,跟堆一样,被所有的线程共享。方法区包含所有的class和static变量。
2.方法区中包含的都是在整个程序中永远唯一的元素,如class,static变量。
AppMain.java
public class AppMain { //运行时, jvm 把Appmain的信息都放入方法区
public static void main(String[] args) { //main 方法本身放入方法区。
Sample test1 = new Sample( " 测试1 " ); //test1是引用,所以放到栈区里, Sample是自定义对象应该放到堆里面
Sample test2 = new Sample( " 测试2 " );
test1.printName();
test2.printName();
}
} Sample.java
public class Sample { //运行时, jvm 把Sample的信息都放入方法区
/** 范例名称 */
private String name; //new Sample实例后, name 引用放入栈区里, name 对象放入堆里
/** 构造方法 */
public Sample(String name) {
this .name = name;
}
/** 输出 */
public void printName() { //print方法本身放入 方法区里。
System.out.println(name);
}
}
堆内存(heap memory)与栈内存(stack memory)的区别
1、heap用在一个application的很多地方,但stack memory只用于一个线程的执行。前者是对象级别,后者是线程级别。
2、只要是对象的创建,都是被存储到heap space中,同时stack中有这个对象的引用地址。stack memory中只包含基本类型变量和存储在heap space中的对象的引用变量。
3、存储在heap中的对象是全局都可以访问的,然而stack memory不能被其他线程访问。
4、stack 的内存管理是使用LIFO的,然而heap的内存管理要更复杂,因为heap是被全局使用的。因此heap memory被分为Young-Generation,Old-Generation。
5、stack memory是短命的,然而heap memory则是application运行的整个生命周期都有他,直到application结束掉。
6、我们使用-Xms和-Xmx jam 参数来定义heap memory的启动size和最大size。stack memory的size则是使用-Xss。
7、当stack memory满了,那么Java runtime就会抛出一个java.lang.*Error的异常。当heap memory 满了,则会抛出java.lang.OutOfMemoryError: Java Heap Space error。
8、stack memory相比heap memory来说是非常小的。另外由于内存分配比较简单(LIFO),stack memory相比heap memory来说是非常快的。