JVM的 GC机制和内存管理

时间:2023-10-14 18:43:01

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来说是非常快的。