本文乃个人感悟总结,希望对大家有所帮助,宗旨为初学者快速了解jvm,为有jvm基础者更好的复习回忆。后期还会出其他文集敬请期待。
一个程序的运行离不开JVM,所以我们从JVM的启动开始讲起,JVM是如何启动的呢,首先自然是装载配置(根据当前路径和系统版本找到JVM.cfg),然后根据配置文件寻找JVM.dll(jvm.dll是动态资源链接库,说白了就是JVM的主要实现),接着就是根据dll初始化JVM获得JNIEvement接口(findclass等操作就是通过这个接口实现的,要实现混合编程也少不了这个接口,什么是混合编程,实际开发过程中单一的Java语言可能满足不了我们的需求,这个时候我们可以使用其他的编程语言,只要其他编程语言的编译结果是一个有效的字节码文件,JVM就能有效的支持这种语言,使得Java能够与其他语言进行混合编程),最后找到main方法并运行,我们的JVM就启动了。JVM启动后我们就接着讲讲一个程序是如何通过jvm运行的整个流程,程序是源码,所以首先我们要将源码通过前端编译器javac编译成字节码文件(这个过程包括词法分析语法分析语义分析最后编译成class文件),然后jvm通过类加载器将字节码文件从本地或者网络传输加载到内存(加载的过程为:加载,验证,准备,解析,初始化,最终形成jvm直接使用的Java类型。同时加载器类型分又为启动类加载器,扩展类加载器,应用程序类加载器,自定义类加载器。它们自上往下查找类是否被加载,自下往上加载类,根据双亲委派模型顶层类加载器不能加载底层类加载器的对象),加载到内存后字节码的通过JVM的JIT(即时编译器,分为两种,client compare和server compare 简称C1和C2)来将字节码编译成机器指令最后由jvm执行引擎来执行。说到我们的class文件被加载到内存,我们就不得不说一下JVM的内存空间了,jvm的内存空间分为方法区、堆、栈、本地方法栈(方法区保存装载的类信息,常量池,静态数据等,同时被所有线程共享。堆保存类对象,线程共享。栈存放方法的局部变量,基本操作数栈,常量池指针,线程私有),而堆进一步细分的话,又分为新生代(Eden空间,幸存空间(From Survivor空间和To Survivor空间))和老年代,而且堆是gc的主要工作空间。所以接下来我们讲讲这个GC了,GC是用来垃圾收集管理内存的,它如何对垃圾进行收集呢,gc内部提供四种垃圾回收的算法(引用计数法、标记清除法、标记压缩法、复制法)。最后我们谈谈如何优化jvm,最牛逼的就是根据项目的情况重写一个jvm啦,例如taobaoVM,或者使用命令行(jps、jstat、jmap、jhat、jstack、jinfo等)查看内存各个区域的占用情况设置参数(-Xmm、-Xms、-Xmn、newRatio等)来优化jvm。