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