GC日志分析
打印GC日志方法,在JVM参数里增加参数,%t 代表时间
java
java -jar ‐Xloggc:./gc‐%t.log ‐XX:+PrintGCDetails ‐XX:+PrintGCDateStamps ‐XX:+PrintGCTimeStamps ‐XX:+PrintGCCause
‐XX:+UseGCLogFileRotation ‐XX:NumberOfGCLogFiles=10 ‐XX:GCLogFileSize=100M
Paralle日志
通过上述日志可以看到JVM配置参数及具体的GC运行情况,可以看到使用了Paralle垃圾收集器。
GC (Allocation Failure]:表示本次为minor gc,括号里的表示GC原因(分配失败)。
[PSYoungGen:65536K->3872K(76288K)] 65536K->3888K(251392K):从该数据可以得到当前年轻代为65536K,GC后为3872K,其中总年轻代大小为76288K,中括号后面的表示堆的总大小。
从图中可以明显看到Full GC被执行了多次,原因是方法区空间不足后,JVM动态扩容。此时建议优化方法区的初始大小(默认为21M)
建议:一般需要将堆的最大最小值和方法区的最大最小值设置上且相同,防止扩容影响性能。
CMS日志
可以看到CMS执行的各个阶段,但不建议直接查看这些日志文件,要借助可视化工具。
G1日志
JVM汇总查看命令
java -XX:+PrintFlagsInitial:表示打印出所有参数选项的默认值
java -XX:+PrintFlagsFinal:表示打印出所有参数选项在运行程序时生效的值
常量池
Class常量池与运行时常量池
Class常量池可以理解为是Class文件中的资源仓库。Class文件中除了包含类的版本、字段、方法、接口等描述信息外,还要一项信息就是常量池,用于存放编译生成的各种字面量和符号引用。
字面量
字面量就是指由字母、数字等构成的字符串或者数值常量
字面量只可以右值出现,所谓右值是指等号右边的值,如:int a=1 这里的a为左值,1为右值。在这个例子中1就是字面量。
java
int a = 1;
int b = 2;
int c = "abcdefg";
符号引用
符号引用是编译原理中的概念,是相对于直接引用来说的。主要包括了以下三类常量:
- 类和接口的全限定名
- 字段的名称和描述符
- 方法的名称和描述符
上面的a,b就是字段名称,就是一种符号引用,还有Math类常量池里的Lcom/tuling/jvm/Math 是类的全限定名, main和compute是方法名称,()是一种UTF8格式的描述符,这些都是符号引用。
这些常量池现在是静态信息,只有到运行时被加载到内存后,这些符号才有对应的内存地址信息,这些常量池一旦被装 入内存就变成运行时常量池,对应的符号引用在程序加载或运行时会被转变为被加载到内存区域的代码的直接引用,也 就是我们说的动态链接了。例如,compute()这个符号引用在运行时就会被转变为compute()方法具体代码在内存中的 地址,主要通过对象头里的类型指针去转换直接引用。
字符串常量池
位置
Jdk1.6及之前: 有永久代, 运行时常量池在永久代,运行时常量池包含字符串常量池
Jdk1.7:有永久代,但已经逐步"去永久代",字符串常量池从永久代里的运行时常量池分离到堆里
Jdk1.8及之后: 无永久代,运行时常量池在元空间,字符串常量池里依然在堆里
设计思想
- 字符串的分配,和其他的对象分配一样,耗费高昂的时间与空间代价,作为最基础的数据类型,大量频繁的创建 字符串,极大程度地影响程序的性能
- JVM为了提高性能和减少内存开销,在实例化字符串常量的时候进行了一些优化
- 为字符串开辟一个字符串常量池,类似于缓存区
- 创建字符串常量时,首先查询字符串常量池是否存在该字符串
- 存在该字符串,返回引用实例,不存在,实例化该字符串并放入池中
三种字符串操作(jdk1.7及以上版本)
直接赋值字符串
String s = "sss"; // s指向常量池中的引用
这种方式创建的字符串对象,只会在常量池中。
因为有"sss"这个字面量,创建对象s的时候,JVM会先去常量池中通过 equals(key) 方法,判断是否有相同的对象
如果有,则直接返回该对象在常量池中的引用;
如果没有,则会在常量池中创建一个新对象,再返回引用。
new String()
String s1 = new String("sss"); // s1指向内存中的对象引用
这种方式会保证字符串常量池和堆中都有这个对象,没有就创建,最后返回堆内存中的对象引用。
步骤大致如下:
因为有"sss"这个字面量,所以会先检查字符串常量池中是否存在字符串"sss"
不存在,先在字符串常量池里创建一个字符串对象;再去内存中创建一个字符串对象"sss";
存在的话,就直接去堆内存中创建一个字符串对象"sss";
最后,将内存中的引用返回。
注:Stringbuilder亦是如此
intern方法
String s1 = new String("zhuge"); String s2 = s1.intern(); System.out.println(s1 == s2); //false
String中的intern方法是一个 native 的方法,当调用 intern方法时,如果池已经包含一个等于此String对象的字符串(用equals(oject)方法确定),则返回池中的字符串。否则,将intern返回的引用指向当前字符串 s1(jdk1.6版本需要将 s1 复制到字符串常量池里)。
小问:如下代码生成了多少对象?结果是?
java
String s1 = new String("he") + new String("llo");
String s2 = s1.intern();
System.out.println(s1 == s2);
// 在 JDK 1.6 下输出是 false,创建了 6 个对象
// 在 JDK 1.7 及以上的版本输出是 true,创建了 5 个对象
分析图:
两个重要等价
java
String s = "a" + "b" + "c"; //就等价于String s = "abc";
String a = "a"; String b = "b";
String c = "c";
String s1 = a + b + c; //就等价于StringBuilder.append()
八种基本类型的包装类和对象池
java中基本类型的包装类的大部分都实现了常量池技术(严格来说应该叫对象池,在堆上),这些类是Byte,Short,Integer,Long,Character,Boolean,另外两种浮点数类型的包装类则没有实现。另外Byte,Short,Integer,Long,Character这5种整型的包装类也只是在对应值小于等于127时才可使用对象池,也即对象不负责创建和管理大于127的这些类的对象。因为一般这种比较小的数用到的概率相对较大。