博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
小白之java虚拟机工作学习面试准备(持续更新)
阅读量:3898 次
发布时间:2019-05-23

本文共 4994 字,大约阅读时间需要 16 分钟。

首先是学习的资源:

1.周志明的《深入理解java虚拟机》
2.b站视频:
3.方志明的专栏:
4.博主:纯洁的微笑:
个人的学习方法就是,先看视频有一个大致的学习框架再看技术博客,然后一点一点不清楚的知识点再去网上找对应的解析,而书则作为巩固用
在这里插入图片描述
目前就是学了这些:
首先要知道的是,JVM(java虚拟机)到底是个什么东西呢,在看了书之后,姑且认为是:

JVM就是我们常说的java虚拟机,它是整个java实现跨平台的最核心的部分,所有的java程序会首先被编译为.class的类文件,这种类文件可以在虚拟机上执行。也就是说class并不直接与机器的操作系统相对应,而是经过虚拟机间接与操作系统交互,由虚拟机将程序解释给本地系统执行。当然只有JVM还不能成class的执行,因为在解释class的时候JVM需要调用解释所需要的类库lib,而jre包含lib类库。

JVM屏蔽了与具体操作系统平台相关的信息,使得Java程序只需生成在Java虚拟机上运行的目标代码(字节码),就可以在多种平台上不加修改地运行。

也就是说,是一个程序,用来将编写好的代码,装换位class类文件,并且存储在这个地方,让操作系统通过虚拟机这个平台来和程序进行交互运行。

总的来说有三个部分
1.类加载器
2.运行时数据区
3.执行引擎
先了解一下java的内存模型:
在这里插入图片描述
再介绍一下每一个运行时数据区里的功能:

线程私有的部分:

1.程序计数器:相当于是用来安排命令的,这需要机组的知识了,也就是用解释器读取载入内存的字节码,并且按照顺序读取字节码中的指令,读取完指令之后,就把他翻译成对应的操作,再按照操作分支、循环、跳转等流程。是唯一一个在java虚拟机规范中没有规定任何OutOfMemoryError的区域

2.虚拟机栈:也被称为方法执行的内存区,因为每个方法从开始到结束都会在里面存入一个栈帧(就和做一个独一无二的标记一样),在栈帧里存一系列信息,每一个方法被调用直至执行完成的过程,就对应着一个栈帧在虚拟机栈中从入栈到出栈的过程

3.本地方法栈:和虚拟机栈一样,也是对应方法执行的内存区,不过这个方法是从本地调用的,也存在栈帧什么的吧,然后也是从入栈到出栈,是一个方法从调用到结束的所有过程。

线程共享的部分:

4.堆:首先他是对象分配内存的区域,所有他会是垃圾回收的主战场,和栈作为对比的就是,栈是对象的引用(也就是调用,和电脑的快捷方式一样),而堆则是实际存对象的地方。姑且认为就是分配内存的地方,之后到垃圾回收部分,还会细说,需要注意的是他也不是连续分配的内存,而是逻辑上联系的

5.方法区和运行时常量池(String在的地方):他有一些便于理解但是不官方的说法,就是他是“非堆”“永久代”,是用来存一些类信息,常量和静态变量,以及编译后的代码这些数据的,由于会存数据,所以他也是会需要垃圾回收的。

新的思维导图是这样的:

在这里插入图片描述

再者开始讲一下双亲委派机制:

我又在这篇博客中提到过:
之后,就要开始讲重要的垃圾回收机制了。
首先要知道的是,对于垃圾的回收都是在线程共享区,也就是java的堆和方法区,并且绝大部分是在堆中的。
为此,需要详细介绍一下java的内存中的堆
在这里插入图片描述
这就是java8之后是堆的形态。
主要就是青年代(里面还分为Eden和Survivior 也就是俗称的伊甸区和幸存区)还有老年区,当然这样划分也是为了更好的垃圾回收。
介绍完了之后,就要重点介绍那几个垃圾回收算法:
标记清除算法:

这个是最基础的算法,就和名字一样,主要的工作就是对一个个内存进行标记是否需要清除,等全都标记完了之后,就开始一块的清除。

在这里插入图片描述

这个算法,可想而知。在效率上:每次进行需要遍历整个内存环境,效率实在是不高。其次,在空间上,清除之后会存在很多的内部碎片(内部碎片的缺点就是在于,之后需要分配较大且连续的内存时,可能会出现总内存够但是依然分配不出内存的情况),可能相对来说比较好的就是在于,这种清理方式不需要移动对象。

标记-复制算法(也被称为复制算法和半区算法)

大致意思就是,首先将堆分类两份,其中一份用来分配内存。当这个内存被分配完了之后,就将所有的还存活的对象移动到未被分配过的内存中,(这里的分配内存也就是对指针的移动)再把这个之前的内存一次性清理完全。他有很多优点,首先就是减少了大量的时间上的开销,也没有内部碎片的担心,但是缺点也是显而易见的,那就是太浪费内存了!
在这里插入图片描述
所以为此,进行了一定的优化,主要是在于新生代中的对象每次回收都基本上只有10%左右的对象存活,所以需要复制的对象很少,效率还不错,所以将内存分配成了一个较大的Eden区和两个survivor(一个是from 一个是to)
好像那些书里都没具体讲这样子的具体步骤,在这里找到了:
首先是将 Eden survivor from survivor to按照8:1:1分
在这里插入图片描述
而具体的步骤就是:
1.进行一次垃圾收集,垃圾收集器会将Eden区存活的对象复制到from survivor中,存活对象计数加一
2.再次进行垃圾回收,将Eden区和from survivor取存活的对象复制到to survivor,此时将to survivor变成from ,将原来的from变成 to(空的是to),存活对象计数加一
3.重复操作。

标记-压缩算法

这个一定程度上类似于标记复制算法,但是使用的场合不太一样,对于那些存活率较高的内存来说,标记并且复制是会被复制很多次的,并且这些复制都是没有必要的,特别是在老年区(因为老年区就是那些一直存活着的内存)。
在这里插入图片描述
压缩也被称为是整理,所以少了复制的操作,而多了整理的操作,也就是将存活的内存向着一端移动。

“方法是死的,人是活的”所以在这些算法都被发明出来的基础之上,提出了分代收集算法,首先将堆分为老年代和新生代,对于新生代这种,存活率不高的,就使用标记复制算法来尽可能的筛选掉那些存活率低的,而对于老年代这些存活率极高的来说,就可以使用标记-整理或者标记清除算法来实现操作。

对于内存,这里还有一点需要提一下的就是关于这个元空间的知识,感觉也是经常被提到的。在jdk8之后,jvm删除了永久代的概念,而是选择了元空间,他也是在方法区中的,他和永久代最大区别在于,元空间是不在虚拟机中,而使用的本地内存,因此是可以通过一些参数来实现对元空间的配置的。

又开始更新了,除了上面的知识点之外,还需要看的是 JVM的性能监控与故障定位以及JVM 的调优

大概是从这几个工具来看
在这里插入图片描述
这些工具都在jdk的lib下的tools.jar下可以查看到
在这里插入图片描述
首先第一个是jps
(这些命令都是可以直接在idea的终端里操作的)
例如:
在这里插入图片描述
jps:JVM Process Status Tool,显示指定系统内所有的HotSpot虚拟机进程。
一般就是有四个参数

-l : 输出主类全名或jar路径

-q : 只输出LVMID

-m : 输出JVM启动时传递给main()的参数

-v : 输出JVM启动时显示指定的JVM参数

再是jstat是看虚拟机运行时状态信息的命令,可以显示出虚拟机进程中的类装载,内存,垃圾收集等

他的参数一共有四个
jstat [option]LVMID [interval][count]
后面两个表示查询间隔和次数,一般可以省略,也就是默认是查询一次
然后LVMID是虚拟机进程的id,可以通过jps来查看
重点就是option,选项众多:
在这里插入图片描述

-class:

是看监视器的装载,卸载数量和总空间以及耗费的时间
在这里插入图片描述

-compiler

是输出即时编译器编译过的方法和耗时
在这里插入图片描述
-gc
是java堆垃圾回收的行为统计
在这里插入图片描述
-gccapacity
和gc相同,但是还会输出java堆各区域用到最大,最小的空间
在这里插入图片描述
-gcutil
是输出已使用空间占总空间的百分比
在这里插入图片描述
-gccause
垃圾收集统计概述(同-gcutil),附加最近两次垃圾回收事件的原因
在这里插入图片描述
-gcnew
统计新生代的行为
在这里插入图片描述
等等 就不一一赘述了

再是jmap,是用来生成heap dump文件的

jmap [option]LVMID
有以下参数:

dump : 生成堆转储快照

finalizerinfo : 显示在F-Queue队列等待Finalizer线程执行finalizer方法的对象

heap : 显示Java堆详细信息

histo : 显示堆中对象的统计信息

permstat : to print permanent generation statistics

F : 当-dump没有响应时,强制生成dump快照

再就是jhat了,他是和jmap搭配使用,用来分析jmap生成的dump

jhat [dumpfile]

-stack false|true 关闭对象分配调用栈跟踪(tracking object allocation call stack)。 如果分配位置信息在堆转储中不可用. 则必须将此标志设置为 false. 默认值为 true.>

-refs false|true 关闭对象引用跟踪(tracking of references to objects)。 默认值为 true. 默认情况下, 返回的指针是指向其他特定对象的对象,如反向链接或输入引用(referrers or incoming

references), 会统计/计算堆中的所有对象。>

-port port-number 设置 jhat HTTP server 的端口号. 默认值 7000.>

-exclude exclude-file 指定对象查询时需要排除的数据成员列表文件(a file that lists data members that should be excluded from the reachable objects query)。 例如,

如果文件列列出了 java.lang.String.value , 那么当从某个特定对象 Object o 计算可达的对象列表时,
引用路径涉及 java.lang.String.value 的都会被排除。>

-baseline exclude-file 指定一个基准堆转储(baseline heap dump)。 在两个 heap dumps 中有相同 object ID 的对象会被标记为不是新的(marked as not being new). 其他对象被标记为新的(new).

在比较两个不同的堆转储时很有用.>

-debug int 设置 debug 级别. 0 表示不输出调试信息。 值越大则表示输出更详细的 debug 信息.>

-version 启动后只显示版本信息就退出>

-J< flag > 因为 jhat 命令实际上会启动一个JVM来执行, 通过 -J 可以在启动JVM时传入一些启动参数. 例如, -J-Xmx512m 则指定运行 jhat 的Java虚拟机使用的最大堆内存为 512 MB. 如果需要使用多个JVM启动参数,则传入多个 -Jxxxxxx.

还有jstack.也是用于生成当前时刻的线程快照

jstack [option]LVMID

-F : 当正常输出请求不被响应时,强制输出线程堆栈

-l : 除堆栈外,显示关于锁的附加信息

-m : 如果调用到本地方法的话,可以显示C/C++的堆栈

最后就是jinfo

他是实时查看和调整虚拟机运行参数
jinfo [option] [args]LVMID

-flag : 输出指定args参数的值

-flags : 不需要args参数,输出所有JVM参数的值

-sysprops : 输出系统属性,等同于System.getProperties()

除了这些以外,还是的就是

VisualVM :在jdk/bin目录下面双击jvisualvm.exe既可使用,启动起来后和jconsole 一样同样可以选择本地和远程,如果需要监控远程同样需要配置相关参数
在这里插入图片描述
和jconsole 直接在jdk/bin目录下点击jconsole.exe即可启动
在这里插入图片描述
最后总结,jvm的学习(针对面试)差不多就告一段落了
在这里插入图片描述

转载地址:http://yifen.baihongyu.com/

你可能感兴趣的文章
Linux之grub.conf的内容介绍
查看>>
网址之Curl API整理说明
查看>>
Python之ftp的用法整理
查看>>
MStar之公司简介
查看>>
STB之业务架构图
查看>>
WebKit之Port篇幅介绍
查看>>
WebKit之Binding案例(testCallback.idl)
查看>>
WebKit之binding分析案例(testInterface.idl)
查看>>
WebKit之binding案例分析(testMediaQueryListListener.idl)
查看>>
Webkit之generate-bindings.pl源码分析
查看>>
WebKit之CodeGenerate-JS的perl脚本的分析和学习
查看>>
Linux之死锁的代码体验
查看>>
WebKit之webIDL详解
查看>>
WebKit之创建Event的2种方法
查看>>
CPP之中介者设计模式
查看>>
Event的三个阶段
查看>>
WebKit之MediaPlayer的简单分析
查看>>
WebKit之MediaPlayer之底层调度的源码分析
查看>>
WIFI之系统模块列表介绍
查看>>
C++之数据库处理代理模块
查看>>