JVM调优1- JVM参数及JDK自带的JVM查询工具

1. JVM的运行参数

1.1 三种参数类型
  • 标准参数

    • help
    • -version
  • -X参数 (非标准参数)

    • -Xint
    • -Xcomp
  • -XX参数(使用率比较高, 常用语jvm调优)

    • -XX:newSize
    • -XX:+UseSeriaIGC
1.2 标准参数

-D 设置系统属性: java -Dstr=hello

String str = System.getProperty("str")
println(str)

输出结果: hello
JVM的两个启动模式: server和client
JVM在启动的时候回根据硬件和操作系统自动选择是使用Server还是Client类型的JVM
64位操作系统只有Server类型,没有Client

1.3 非标准参数

JVM的运行模式:
-Xint: 解释模式, 强制要求JVM执行所有的字节码(interpreted mode)
-Xcomp: 编译模式,JVM在第一次使用时会把所有字节码编译成本地代码(compiled mode)
-Xmixed:混合模式,将解释模式和编译模式混合使用, JVM默认的模式, 也是推荐的模式(mixed mode)

1.4 -XX 参数

-XX参数也是非标准参数, 主要用于JVM的调优和debug操作
两种使用方式:

  • boolean类型
    格式: -XX:[+-]<name> 表示启用或者禁用name指令
    -XX:+DisableExplicitGC表示禁用手动调用gc操作, System.gc()无效
  • 非boolean类型
    格式: -XX<name>=<value> 表示name的属性值为value
    -XX:NewRatie=1 表示新生代和老年代的比值
1.5 -Xms, -Xmx

-Xms和-Xmx分别是设置jvm的堆内存的初始大小和最大大小
-Xms512m相当于: -XX:InitialHeapSize=512m
-Xms2048m相当于:-XX:MaxHeapSize=2048m

JVM启动时会自动设置Heap size的值, -Xms初始空间是物理内存的1/64, -Xmx最大值时物理空间的1/4。
进行JVM优化时, 可将-Xms和-Xmx设置值相同, 最大值不超过物理内存的80%

1.6 查看jvm运行参数:

运行java命令时打印参数: -XX:+PrintFlagsFinal
参数列表中: = 表示默认值, :=表示值被修改过

1.7 查看正在运行的进程的jvm信息

jps -l查看当前系统中所有运行的java项目的进程id及详细包信息
jinfo -flags 29200 查看进程id为29200进程的所有jvm参数信息
jinfo -flag MaxheapSize 29200 查看pid下的具体某一jvm信息

image.png

2. jvm的内存模型

jvm的内存模型1.7和1.8有较大的区别

2.1 jdk1.7的堆内存模型:
  • 年轻区: 新new的一些对象会在这个区域,young区被分为:Eden区和两个大小严格相同的Survivor区, 当Eden区变满时, 数据会移到Survior中,几次jvm垃圾收集后, 依然存活的Survivor会知道老年区
  • 老年区:tenured区主要保存生命周期长的对象, 一般是一些老的对象
  • 永久区: 主要保存class, method, field对象, 这部分空间一般不会溢出
2.2 jdk1.8的堆内存模型

年轻区:Eden+2*Survivor
老年区:OldGen
元数据空间(Metaspace):Matespace是不是在虚拟机内部, 而是占用服务器的内存空间,这是和1.7最大的区别

2.3 为什么废除了1.7的永久区

现实中是因为永久代内存总会发生不够用或者内存泄漏, 基于此将永久区废弃, 改为使用本地的内存空间

2.4 通过jstat命令进行查看堆内存的使用情况

jstat 参数指令 进程id 时间间隔 查询次数
jstat -class 29200 查看类加载情况
loaded: 加载类的数量
Bytes:类占得空间
uloaded:未加载类的数量
Time: 加载占用的时间

image.png

jstat -compiler 29200 查看编译情况

image.png

查看垃圾回收器的使用情况
jstat -gc 29200 1000 10 每1000毫秒(1秒)钟打印一次gc使用情况, 总共打印10次

image.png

3. jmap的使用以及内存溢出分析

3.1 查看内存使用情况

jmap -heap 29200

3.2 查看内存中对象数量及大小:

jmap -histo <pid> | more 查看所有对象的
jmap -histo:live <pid> | more 查看活跃对象的

image.png

[B-> byte
[I -> int
[C->char

3.3 将内存使用情况dump到文件中, jhat对快照文件分析

jmap -dump:format=b,file=filename <pid>
例如 jmap -dump:format=b,file=dumptest.dat 29200 会生成一份29200内存使用情况的二进制文件dumptest.dat
使用jhat对dump的二进制文件分析:jhat port 9999 dumptest.dat

image.png

浏览器访问7000端口:
image.png

4. 内存溢出的定位与分析

内存溢出在生产环境中经常会遇到, 比如不断地将数据写入到一个集合中,出现了死循环, 读取超大文件等等,都可能造成内存溢出。

4.1模拟内存溢出:

设置jvm参数: -Xms8m -Xmx8m -XX:+HeapDumpOnOutOfMemoryError
jvm初始内存8m, 最大内存8m, 内存溢出时dump内存快照
编写代码:

    public static void main(String[] args) {
        ArrayList<String> strings = new ArrayList<>();
        for (int i = 0; i <= 1000000000; i++){
            String id = "";
            for (int j = 0; j < 1000; j++){
                id += UUID.randomUUID().toString();
            }
            strings.add(id);
        }
    }

mat工具分析内存


image.png
4.2 jstack的使用:

有时候我们需要查看jvm中的线程执行情况, 比如发现CPU的负载突然增高,出现了死锁,死循环等, 由于程序正常运行的,没有任何的输出, 从日志方面也看不出什么问题, 需要从jvm的内部线程的执行情况查找并且分析原因。
jstack <pid> 可以看到当前pid进程中所有线程的执行情况。

5. jstack中Java中线程的状态

  1. 初始状态(NEW): 创建一个thread对象, 但还未调用start启动线程, 则处于初始状态
  2. 运行状态(RUNNABLE):
    • 就绪状态: 等待CPU分配执行权, 放在就绪队列中
    • 运行状态:获得CPU的执行权, 一个CPU在同一时间只能执行一个线程,所以每个CPU在每个时刻都只有一条运行态的线程。
  3. 阻塞状态(BLOCKED):java中指请求某一锁失败时, 线程会进入阻塞态, 阻塞态会不断地请求资源, 一旦请求成功就会进入就绪队列, 等待CPU分配。
  4. 等待状态(WAITING):无线等待, 当线程中调用 wait, join, park等函数时,线程进入等待状态,等待线程会释放CPU以及锁资源,进入等待队列, 需要其他线程指示才能继续运行。
  5. 超时等待态 (TIMED_WAITING): 有限等待,与等待态的区别是, 到了超时时间会进入阻塞队列, 开始竞争锁。
  6. 终止态
5.1死锁问题

模拟死锁问题:

   private static Object obj1 = new Object();

    private static Object obj2 = new Object();


    public static void main(String[] args) {
        new Thread(new Thread1()).start();
        new Thread(new Thread2()).start();
    }

    private static class Thread1 implements Runnable{
        @Override
        public void run() {
            synchronized (obj1){
                System.out.println("Thread1 拿到了 obj1 的锁!");

                try {
                    // 停顿2秒的意义在于,让Thread2线程拿到obj2的锁
                    Thread.sleep(2000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }

                synchronized (obj2){
                    System.out.println("Thread1 拿到了 obj2 的锁!");
                }
            }
        }
    }

    private static class Thread2 implements Runnable{
        @Override
        public void run() {
            synchronized (obj2){
                System.out.println("Thread2 拿到了 obj2 的锁!");

                try {
                    // 停顿2秒的意义在于,让Thread1线程拿到obj1的锁
                    Thread.sleep(2000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }

                synchronized (obj1){
                    System.out.println("Thread2 拿到了 obj1 的锁!");
                }
            }
        }
    }

线程1和线程2相互持续持有对方需要的锁资源, 造成死锁问题。
使用jstack查看死锁进程:


image.png

image.png

jstack查找除了发生死锁进程的原因, 以及对应的堆栈信息。

6. 使用JDK子代的 Java VisualVm

综合的结合了以上介绍jvm性能查看的jdk自带可视化工具:

位置: 当前jdk安装目录bin下的jvisualvm
image.png

既可以查看本地java进程, 也可以查看远程进程

配置远程java进程如tomcat的启动文件catalina.sh, 允许jmx远程监控

JAVA_OPTS="
-Dcom.sun.management.jmxremote 
-Dcom.sun.management.jmxremote.port=9999 
-Dcom.sun.management.jmxremote.authenticate=false 
-Dcom.sun.management.jmxremote.ssl=false"

开放9999端口, 不需要认证,关闭ssl认证

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,457评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,837评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,696评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,183评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,057评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,105评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,520评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,211评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,482评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,574评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,353评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,213评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,576评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,897评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,174评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,489评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,683评论 2 335

推荐阅读更多精彩内容