回炉重造java----JVM

为什么要使用JVM

①一次编写,到处运行,jvm屏蔽字节码与底层的操作差异

②自动内存管理,垃圾回收功能

③数组下边越界检查

④多态

JDK,JRE,JVM的关系

13466d21ae7e49bca0f837aaa6b966b1.png

JVM组成部分

755fc9b64657438fbb494035609733ac.png

JVM的内存结构

cc0ca0d7bc6f434b9c6fc647c21be397.png

《一》程序计数器(PC Register)

作用:

java代码执行流程: java源代码-->二进制字节码文件(jvm指令)-->解释器-->机器码-->cpu执行,而程序计数器就是去记住下一条jvm指令的执行地址。

特点:

        ①线程私有

        ②唯一不会存在内存溢出的区

《二》虚拟机栈(JVM Stack)

概念:每个线程运行时所需要的内存,称为虚拟机栈。

555fb117b7674456b602655afdd50891.png

21181c26352d40aba464d58945044c44.png

        1. 答:垃圾回收不涉及栈内存,因为栈中是运行方法时的存放的地方,当方法结束后栈帧会自动退栈释放空间,不需要垃圾回收机制来处理。垃圾回收主要是去处理堆内存。

        2. 答:不是,内存分配越大并不会对程序的执行效率有提升,反而会减少内存空间和线程的数量。

        3. 答:方法内的局部变量是安全的,因为一个线程对应自己的一个栈,局部变量各自存在栈中。但是如果是static修饰的变量线程间就会共享,可能会产生线程安全问题。另外如果局部变量是引用类型并且逃离了方法的作用范围,就会产生线程安全问题。

栈内存溢出:java.lang.StackOverflowError

        ①方法(栈帧)过多,栈帧内存超过栈的总内存就会发生栈溢出 

        ②栈帧过大直接挤满了栈发生栈溢出(反复调用)

 线程诊断:

        ①top命令可以查看所有正在运行的进行和系统负荷提供不断更新的概览信息,包括系统负载、CPU利用分布情况、内存使用、每个进程的资源占用情况等信息

        ②ps H -eo pid,tid,%cpu |  grep 进程ID    查看线程的信息

        ③jstack 进程ID  查看进程的线程信息

        

《三》本地方法栈(Native Method Stacks)

        有的时候我们需要调用操作系统底层的一些方法,但是java语言不可以直接调用,所以就需要一些更底层的c或者c++写的本地方法去间接调用。所以我们要创建一个本地方法栈去存储本地方法运行时需要的空间

        

《四》堆(Heap)

        ①通过new 创建的对象都会使用堆内存

        ②线程共享,其中的对象都需要考虑线程安全的问题

        ③垃圾回收机制

堆内存溢出: java.lang.OutOfMemoryError

堆内存诊断:

        ①jps   查看当前正在运行的java进程

        ②jmap -heap 进程号   查看堆内存的占用情况

        ③jconsole

《五》方法区(Method Area)

        ①线程共享,在虚拟机启动时创建,用于存储已被Java虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等数据

        ②ClassLoader,类加载器,加载类的二进制字节码

18b7358fdb574ce7ab097310fe1bee03.png

 方法区内存溢出:

635af3f332ae49288badd1ad4a4b3066.png

运行时常量池 

①二进制字节码文件:类的基本信息+常量池+类方法定义包括虚拟机指令

②常量池:是.class文件中的,其实就是一张表,虚拟机指令根据常量池去找到要执行对应的类名,方法名,参数等信息

③运行时常量池:当类被加载时他的常量池信息就会放进运行时常量池,并且把其中的符号地址编程真实地址

StringTable串池(hashtable结构,不能扩容)

ed60647482e6458db5e6b4fffbe734bb.png

常量池被加载到运行时常量池的时候,这时 a b ab 都是常量池中的符号,还不是java对象。当程序中要使用的时候会先去查看串池中有没有,没有的话就把这些符号放到串池当中变成字符串对象。这个是叫做字符串的延迟加载,不用就不加载

878042611e8144a097c859ab89a47949.png

s1+s2会在堆中创建一个新的字符串对象。其底层是一个StringBuilder对象,使用toString方法最后再new String对象,注意:这个对象是只在堆中,不在串池中

s5中,两个字符串拼接起来是直接在串池中找

注意,1.6版本的话如果串池中没有的话,会复制一份对象放进去,而不是放本身,所以串池中的对象跟外面的那个不是一个,外面那个对象还是在堆中

来个题目练练手:

590998b934964eeba976a67595418752.png

StringTable的位置:1.6下存放在常量池(永久代)中,1.7之后就放在堆中,因为在常量池中的回收效率很低 

4adc39b93c144a14a0bb22028033f04c.png

StringTable垃圾回收

当存入的数据大于Stringtable的空间大小时,会触发垃圾回收机制

StringTable性能调优

①参数StringTableSize来调整StringTable的桶个数,让下面的链表不能太长

②当有大量的字符串对象时,考虑将字符串对象是否入池,去重优化字符串的个数

《六》垃圾回收

dce5b6284a8d43cea90c0f3514fe41b6.png

1,如何判断对象可以回收:

        ①引用计数法

        ②可达性分析算法,这个算法的基本思想就是通过一系列的称为“GC Roots”的对象作为起始点,从这些节点开始向下搜索,搜索所走过的路径称为引用链,当一个对象到 GC Roots 没有任何引用链相连时,则证明此对象是不可用的。

4dfb6d38b081432d920e9b711cccf35a.png

        ③Java的五种引用

9de0ff37a4eb4ab086124d85bdb86e43.png

2,垃圾回收算法

标记—清除:分为两个阶段,一个是标记阶段,为每个对象进行标记看是不是被GC ROOT引用;第二个阶段是清除阶段,该阶段对死亡的对象进行清除,执行 GC 操作。

        优点:是可以解决循环引用的问题必要时才回收(内存不足时)

        缺点:标记和清除的效率不高,尤其是要扫描的对象比较多的时候,会造成内存碎片

标记—整理:也是分为两个阶段,一个是标记阶段,为每个对象进行标记看是不是被GC ROOT引用;第二个阶段是整理,在清理垃圾的过程中将所有存活的对象整理移动一下,再去清理空间

        优点:解决标记清除算法出现的内存碎片问题。

        缺点:压缩阶段,由于移动了可用对象,需要去更新引用。

复制算法:将内存分成两个相同大小的区,分别为from和to。使用时只使用一个,当进行垃圾回收时,首先将from区的垃圾标记,然后把存活的对象全部放去to区,再清空from区,执行完之后把二者身份互换。

        优点:能解决内存碎片

        缺点::会造成一部分的内存浪费;如果存活对象的数量比较大,复制算法的性能会变得很差。

3,分代垃圾回收

但是在jvm中不可能只使用上面算法的一种,而是是结合上面的算法,形成一种分代垃圾回收算法

511cd9c898704a1bb98b9784631601c1.png

  • 堆被划分为新生代和老年代,对象首先被分配在新生代中的伊甸园区
  • 当新生代空间不足时,触发minor gc,将伊甸园和from区中存活的对象复制到to中并且对象年年龄+1,到对象年龄到达一定数值(15)后会被放到老年代中,然后再将死亡的对象回收掉,最后交换from和to区
  • minor时会引发stop the world,暂停其他线程,等待垃圾回收
  • 当老年代空间不足时,先尝试minor gc,如果还不够就会触发full gc,STW时间会更长
  • 大对象,当对象的大小超过新生代的空间,并且minor gc之后也没用,会直接晋升到老年代

4,垃圾回收器        

可分为三大类:串行,吞吐量优先,响应时间优先

9422fc72f7754f80b0fb8a592d5b9fda.png

①串行557f69231f504a1290b7961f1b1372e3.png

②吞吐量优先

4ed5b23d7c6c4aa3abc6f503efcc3ae6.png

③响应时间优先(并发)

19b28c79cd1d47a18c5198e14b80da6f.png

 CMS:低延迟,基于标清除算法,并发。

初始标记:标记GC Root   -  》 并发标记    -   》 最终标记    -  》  并发清理:

 G1垃圾回收器(并发)

dcd1318a88db4927913c7359bb89572d.png

e207b50ff2f04b77b94f3671698f8af8.png

 第一阶段:新生代(young collection):

47051126cdd44c7196baeb5938a4781b.png

第二个阶段:新生代+CM(concurrent mark):

991028a9aae04e8688043631c4450f0c.png

第三个阶段:混合收集(Mixed collection):

616c0a8d66484c2196308f4b18b95bb2.png

对老年代的回收时,当老年代数量较少时,会全部进行复制回收,但数量较多时,为了达到MaxGcPauseMillis的要求,会进行有挑选的选择垃圾多的老年代区进行复制回收。 

跨代引用:卡表技术

75b086017450432fb8e7a1c68cb8aacc.png

 《七》类加载器

31f860a3b15e467c8539500d81365f03.png

 1,类字节码文件结构

魔数+版本+类相关信息+常量池+各种方法的信息(包括构造方法,主方法,属性,参数,成员变量)

2,字节码指令

常量池加载时放进运行时常量池,方法字节码存入到方法区,栈帧由局部变量表+操作数栈了,由执行引擎条条虚拟机指令

public class test {
    public static void main(String[] args) {
        int i =0;
        int j =0;
        while(j<10){
            i = i++;
            j++;
        }
        System.out.println(i);
    }
}

结果输出为0。i每次放进去操作数栈之后,局部变量表中虽然执行了++操作,但是之后操作数栈中的0又会被赋值到i中,一直循环。

多态的原理:

60e3a508771b4baf8d5d00c773f9e0cc.png

异常的原理:

3a5ab05835d9492fbc3250731b6c065f.png

finally块:其底层原理就是把finally块中的代码部分分别放在try的后面和catch的后面,所以说finally中的代码一定会执行。 ①而且就算是没有捕获到的异常,也会被抓到然后执行finally的代码,再抛异常,②而且finally中抛异常和return只会执行一个

1ca31fdf8f9b4e8b8fa02d6b691d8ebb.png

Synchronized原理:

d10900a648344fce993f571f249e7260.png

 3,编译期处理

①默认构造器

03af11eff9ef45e6a98e6d127a9f466b.png

 ②自动拆装箱

22200b032c944a5ab9aece2151ae8346.png

 ③泛型擦除

49c3a74d9a5a4d44a7568ff69d2e803a.png

 ④可变参数->数组

0f08b754475f466eb7cc62cb603c673f.png

 ⑤方法重写

f9375b6fd47646ac8fa9d8195433aa70.png

4,类加载阶段  ----->  加载,链接,初始化

加载 

40b5662fdbef4981b51dcbef5d21f401.png

链接

第一步:验证。检查字节码是否符合JVM的规范        

第二步:准备。为静态变量分配空间,设置默认值(这个阶段只声明,不赋值),而被final修饰的变量在准备阶段会赋值

第三步:解析。将常量池中的符号引用解析为直接引用

初始化

6a791fb18cc14509b38901c2ef2d47a7.jpg

 练习:a,b不会被初始化,因为在链接的准备阶段已经赋值了,而c作为包装类,会拆箱创建对象8cdc30b17e314e59a9f2ac3c7890661c.jpg

5,类加载器(启动,扩展,应用程序,自定义)

8a0b915b1b404b1b988f994beb534ef8.jpg

①启动类加载器 class.getClassLoader = null,相关参数:Xbootclasspath

②双亲委派原理,在加载前先访问上级加载器是否加载

b108940ad11742248fbfb0f9ac4f7998.jpg

6,运行期优化

6.1 逃逸分析,即使用JIT即时编译器

6.2

6.3

《八》JMM(内存模型)

①synchronized,底层由monitorenter和monitorexit指令实现

4f7652ca3cf14052a164711cd0fc85c8.jpg

  • 37
    点赞
  • 17
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值