美文网首页
学习记录

学习记录

作者: Pure_Dream | 来源:发表于2020-07-09 11:19 被阅读0次

    synchronized

    java 中每一个对象都可以作为锁,三种应用方式,修饰:

         1、普通方法  锁是当前实例对象

         2、静态方法  锁是当前类的class对象

         3、代码块 锁是括号里的对象

    修饰方法时,在父类中修饰某个方法,子类覆盖此方法,在默认情况下,该方法不是同步的。如果想做到同步,(1)可以在子类方法处添加synchronized,(2)也可以调用父类的方法

    JVM 类加载

    生命周期:加载-连接(验证、准备、解析)-初始化-使用-卸载

    1、加载:一个jar包在运行的时候,会指定一个main方法,作为入口方法,这个时候jvm会首先将此方法所在类加载到jvm,当代码在运行的时候遇到new的时候,继续将该对象加载到jvm。

        (1)使用new实例化对象时,读取和设置类的静态变量、静态非字面值常量(静态字面值常量除外)时,调用静态方法时。

        (2)对内进行反射调用时。

        (3)当初始化一个类时,如果父类没有进行初始化,需要先初始化父类。

        (4)启动程序所使用的main方法所在类

        (5)当使用1.7的动态语言支持时。

        如上5种场景又被称为主动引用,除此之外的引用称为被动引用,被动引用有如下3种常见情况

             通过子类引用父类的静态字段,只会触发父类的初始化,而不会触发子类的初始化。

             定义对象数组和集合,不会触发该类的初始化

             类A引用类B的static final常量不会导致类B初始化(注意静态常量必须是字面值常量,否则还是会触发B的初始化)

             通过类名获取Class对象,不会触发类的初始化。如System.out.println(Person.class);

             通过Class.forName加载指定类时,如果指定参数initialize为false时,也不会触发类初始化。

             通过ClassLoader默认的loadClass方法,也不会触发初始化动作

        注意:被动引用不会导致类初始化,但不代表类不会经历加载、验证、准备阶段。

    2、连接阶段:

    2.1 验证:确保被加载的类的正确性

        确保Class文件的字节流中包含的信息符合当前虚拟机的要求,并且不会危害虚拟机自身的安全。

    文件格式验证:验证字节流是否符合Class文件格式的规范,如:是否以模数0xCAFEBABE开头、主次版本号是否在当前虚拟机处理范围内等等。

    元数据验证:对字节码描述的信息进行语义分析,以保证其描述的信息符合Java语言规范的要求;如:这个类是否有父类,是否实现了父类的抽象方法,是否重写了父类的final方法,是否继承了被final修饰的类等等。

    字节码验证:通过数据流和控制流分析,确定程序语义是合法的、符合逻辑的,如:操作数栈的数据类型与指令代码序列能配合工作,保证方法中的类型转换有效等等。

    符号引用验证:确保解析动作能正确执行;如:通过符合引用能找到对应的类和方法,符号引用中类、属性、方法的访问性是否能被当前类访问等等。

        验证阶段是非常重要的,但不是必须的。可以采用-Xverify:none参数来关闭大部分的类验证措施。

    2.2 准备:为类的静态变量分配内存,并将其赋默认值

        为类变量分配内存并设置类变量初始值,这些内存都将在方法区中分配。对于该阶段有以下几点需要注意:

    只对static修饰的静态变量进行内存分配、赋默认值(如0、0L、null、false等)。

    对final的静态字面值常量直接赋初值(赋初值不是赋默认值,如果不是字面值静态常量,那么会和静态变量一样赋默认值)。

     2.3 解析:将常量池中的符号引用替换为直接引用(内存地址)的过程

        符号引用就是一组符号来描述目标,可以是任何字面量。属于编译原理方面的概念如:包括类和接口的全限定名、字段的名称和描述符、方法的名称和描述符。

        直接引用就是直接指向目标的指针、相对偏移量或一个间接定位到目标的句柄。如指向方法区某个类的一个指针。

    3、 初始化:为类的静态变量赋初值

      注意:只有对类的主动使用才会导致类的初始化。

    如何打破双亲委派模型:

    1.重写loadclass()方法

    2.spi机制

    新生代什么情况下会晋升为老年代

    1.长期存活的对象进入老年代,对象每熬过一次GC年龄+1,默认年龄阈值15 (-XX:+MaxTenuringThreshold=10)

    2. 对象太大新生代无法容纳直接分配到老年代

    3.eden区满了,进行minor gc之后,edon和一个survior区仍存活的对象无法放到to survior 区,则通过分配担保机制放到年老代

    4.动态年龄判断,为了使内存分配更加灵活,jvm不一定要求对象年龄达到MaxTenuringThreshold(15)阈值才晋升为年老代,若survior区,相同年龄的对象总大小占survior空间的一般,则大于等于这个年龄的对象,则会在minor gc的时候移动到老年代。

    垃圾收集器

    Serial New收集器是针对新生代的收集器,采用的是复制算法

    Parallel New(并行)收集器,新生代采用复制算法,老年代采用标记整理

     Parallel Scavenge(并行)收集器,针对新生代,采用复制收集算法   吞吐量优先

    Serial Old(串行)收集器,新生代采用复制,老年代采用标记整理

    Parallel Old(并行)收集器,针对老年代,标记整理

    CMS收集器,基于标记清理,初始标记,并发标记,重新标记,并发清除,其中初始标记和重新标记会触发stw

    G1收集器:整体上是基于标记清理,局部采用复制

    Parallel Old的出现结合Parallel Scavenge,真正的形成“吞吐量优先”的收集器组合。

    jstat 命令 jstat -gcutil pid

    相关文章

      网友评论

          本文标题:学习记录

          本文链接:https://www.haomeiwen.com/subject/kyyjcktx.html