JVM基础知识(原创)

JVM简介

JVM是Java Virtual Machine(Java虚拟机)的缩写,JVM是一种用于计算设备的规范,它是一个虚构出来的计算机,是通过在实际的计算机上仿真模拟各种计算机功能来实现的。Java虚拟机包括一套字节码指令集、一组寄存器、一个栈、一个垃圾回收堆和一个存储方法域。 JVM屏蔽了与具体操作系统平台相关的信息,使Java程序只需生成在Java虚拟机上运行的目标代码(字节码),就可以在多种平台上不加修改地运行。JVM在执行字节码时,实际上最终还是把字节码解释成具体平台上的机器指令执行。编译虚拟机的指令集与编译微处理器的指令集非常类似。

JDK内存

Java 的内存模型分为
Young(年轻代也叫new区)

Young分为 Eden 区和两个Survivor 区(from和to,这两个Survivor区大小严格一至),新的对象实例总是首先放在Eden 区,Survivor区作为Eden区和Tenure(终生代)的缓冲,可以向 Tenure(终生代)转移活动的对象实例。
Tenured(终身代,也叫old区)

Tenure中存放生命周期长久的实例对象,但并不是如它的名字那样是终生的,里面的对象照样会被回收掉。

Young和Tenure共同组成了堆内存,也就是heap内存或heap区。
Perm(永久代)
有些旧版本也叫作:New Old Perm 叫法不同,表达的意思却是基本相同。Perm则是非堆内存的组成部分。主要存放加载的Class类级对象如class本身,method,field等等。
Virtual区
在JVM启动时,就已经保留了固定的内存空间给Heap内存,这部分内存并不一定都会被JVM使用,但是可以确定的是这部分保留的内存不会被其他进程使用。这部分内存大小由 -Xmx 参数指定。
而另一部分内存在JVM启动时就分配给JVM,作为JVM的初始Heap内存使用。影响这个的参数是 -Xms ,如果 -Xms 指定的值比-Xmx 的小,那么两者的差值就是Virtual内存值。随着程序的运行,Eden区,Tenured区和Perm区会逐渐使用保留的Virtual空间。
如果没有具体指定,初始和最大堆内存将根据机器的内存计算得出。参数DefaultInitialRAMFraction 和 DefaultMaxRAMFraction 会影响最终的结果,如下表所示:
FormulaDefault
initial heap size memory / DefaultInitialRAMFraction memory / 64
maximum heap size MIN(memory / DefaultMaxRAMFraction, 1GB) MIN(memory / 4, 1GB)
可以看到堆内存默认值最大不会超过1G。
JVM会根据堆内存的使用情况自动决定何时扩张和缩减实际堆内存的大小,可以用VM参数 -XX:MinHeapFreeRatio= 和 -XX:MaxHeapFreeRatio= 使用堆内存空闲百分比来定义,一般在32位机器上的默认值如下:
ParameterDefault Value
MinHeapFreeRatio 40
MaxHeapFreeRatio 70
-Xms 3670k
-Xmx 64m
-Xms/-Xmx:
java heap并不是越大越好,对他的一般优化原则是够用的情况下,尽可能的小,因为太大的话会浪费内存,同时影响GC的效率。当空闲堆内存所占堆内存百分比低于40%,JVM就会试图扩张堆内存空间;当空闲堆内存所占堆内存百分比高于70%,JVM就会试图压缩堆内存空间。
ps:以上默认值在不同平台会有不同的值,如果是64位系统,这些值一般需要扩张30%,来容纳在64位系统下变大的对象。
GC

GC是Garbage Collection的缩写,即垃圾回收机制,在Java中开发人员无需使用指针来管理内存,GC是JVM对内存(实际上就是对象)进行管理的方式。
第一种为单线程GC,也是默认的GC。,该GC适用于单CPU机器。
第二种为Throughput GC,是多线程的GC,适用于多CPU,使用大量线程的程序。第二种GC与第一种GC相似,不同在于GC在收集Young区是多线程的,但在Old区和第一种一样,仍然采用单线程。-XX:+UseParallelGC参数启动该GC。
第三种为Concurrent Low Pause GC,类似于第一种,适用于多CPU,并要求缩短因GC造成程序停滞的时间。这种GC可以在Old区的回收同时,运行应用程序。-XX:+UseConcMarkSweepGC参数启动该GC。

第四种为Incremental Low Pause GC,适用于要求缩短因GC造成程序停滞的时间。这种GC可以在Young区回收的同时,回收一部分Old区对象。-Xincgc参数启动该GC。
不管采用什么算法,GC总是会导致应用暂停的,这个时间长短从毫秒到秒之间不等,因此会影响应用的相应时间,多长的停顿在接受范围内取决于应用的特征,可以通过设置GC停顿的时间来调整(注意只是期望的时间,而不是绝对)

相关调整的目标:
短生命周期的对象不要进入Old区
短生命周期的对象在minor GC的时候干掉
长生命周期的对象要放到Old区
长生命周期的对象可以被Full GC清理掉,但是Full GC要调整到尽量少发生

JVM中的ClassLoader

类加载器(class loader)用来加载java类到java虚拟机中。一般来说,java虚拟机使用java类的方式如下:Java源程序(.java文件)在经过java编译器编译之后就被转换成java字节代码(.class文件)。类加载器负责读取Java字节代码,并转换成java.lang.class类的一个实例。每个这样的实例用来表示一个java类。

Java程序是由许多独立的类文件组成的,每个文件对应于一个java类。此外,这些文件并非立即全部加载如内存,而是根据程序需要装入内存。ClassLoader便是JVM中将类装入内存的零件,用户可以定制自己的ClassLoader。JVM中缺省的ClassLoader可以完成将本地已文件系统装入类文件的任务。用户指定的ClassLoader可以拥有JVM缺省的ClassLoader所不具有的功能。例如:用户可以使用自己创建的ClassLoader从非本地硬盘或者从网络装入可执行内容。

java应用环境中不同的class分别由不同的ClassLoader负责加载。
一个jvm中默认的classloader有Bootstrap ClassLoader、Extension ClassLoader、App ClassLoader,分别各司其职:
Bootstrap ClassLoader:用来在JVM启动时加载核心类库,主要是 %JRE_HOME/lib/ 目录下的rt.jar、resources.jar、charsets.jar和class等
Extension ClassLoader:负责加载java扩展类,主要是%JRE_HOME/lib/ext目录下的jar和class
App ClassLoader:负责加载当前java应用classpath变量中的所有类。
其中Bootstrap ClassLoader是JVM级别的,由C++编写;Extension ClassLoader、App ClassLoader都是java类,都继承自URLClassLoader超类。
Bootstrap ClassLoader由JVM启动,然后初始化sun.misc.Launcher ,sun.misc.Launcher初始化Extension ClassLoader、App ClassLoader。
下图是ClassLoader的加载类流程图,以加载一个类的过程类示例说明整个ClassLoader的过程。


Java 类加载器层次结构 
类加载器包含具有父类加载器和子类加载器的层次结构。父类加载器和子类加载器之间的关系类似于超类和子类之间的对象关系。引导类加载器是 Java 类加载器层次结构的根。Java 虚拟机 (JVM) 创建引导类加载器,它将加载 JVM 中包含的 Java Development Kit (JDK) 内部类和 java.* 包。(例如,引导类加载器加载 java.lang.String。)
扩展类加载器是引导类加载器的子类加载器。扩展类加载器加载 JDK 的扩展目录中包含的所有 JAR 文件。这是一种无需在类路径中添加条目即可扩展 JDK 的便捷方法。但扩展目录中的所有内容都必须是自包含的,且只能引用扩展目录中的类或 JDK 类。

系统类路径类加载器扩展 JDK扩展类加载器。系统类路径类加载器加载 JVM 类路径中的类。
加载类 
类加载器在加载类时使用委托模型。类加载器实现首先检查其缓存,查看是否已经加载所请求的类。由于不重复从磁盘中加载类,而是使用该类在缓存内存中的副本,所以,这种类验证可以提高性能。如果在类缓存中找不到该类,则当前类加载器会要求其父类加载器提供该类。仅当父类加载器也无法加载该类时,该类加载器才会尝试加载该类。如果某个类既存在于父类加载器中,又存在于子类加载器中,则将加载父类加载器中的类。遵循这种委托模型可以避免同时加载多份相同的类。加载多份相同的类会引发 ClassCastException。
类加载器会先要求其父加载器加载类,然后再尝试自己加载该类。

参考至:《叱咤风云:WebLogic企业级运维实战》戴冠平著

             http://longdick.iteye.com/blog/442213

             http://songyishan.iteye.com/blog/1135183

             http://hi.baidu.com/love200456/blog/item/9d4d70fbdc38c970024f564f.html

             http://blog.sina.com.cn/s/blog_677e5328010109a3.html
本文原创,转载请注明出处、作者
如有错误,欢迎指正
邮箱:czmcj@163.com

作者:czmmiao  文章出处:http://czmmiao.iteye.com/blog/1614446

时间: 2024-09-21 22:54:41

JVM基础知识(原创)的相关文章

JVM 基础知识(GC)

几年前写过一篇关于JVM调优的文章,前段时间拿出来看了看,又添加了一些东西.突然发现,基础真的很重要.学习的过程是一个由表及里,再由里及表的过程,所谓的"温故而知新".而真正能走完这个轮回的人,也就能称为大牛或专家了.这个过程可能来来回回,这就是所谓"螺旋上升",而每一次轮回都有新的发现.   这回添加的东西主要集中在基础的一些问题上,还有一些这两年思考的问题.这些问题可能平时我们不会刻意去想,但是真正看清楚了,却发现还是大有裨益的,希望对大家都有帮助~ 一.基础概

基于JVM的动态语言Groovy 基础知识汇总

在使用Java的过程中,和C#的语法相比有些还是比较麻烦,比如异常.get set等问题,毕竟Java的发展时间比C#长了很多,很多问题当初设计时没有考虑到,为了向前兼容,不得不保留一定的历史负担(如泛型的处理,java的擦除法实现就是后续的兼容考虑).不过最近在一个项目中使用groovy grails感觉很是方便,特别groovy和java的集成十分的方便. 下面把groovy涉及的一些基础知识整理一下,供使用参考,groovy本身的文档也很全面,但篇幅太长,如下作为一个简明的参考. 官网 h

黑马程序员 一、java 概述与基础知识

获取更多资源关注Java帮帮IT资源分享网 一.黑马程序员-java 概述与基础知识 1.何为编程? 编程就是让计算机为解决某个问题而使用某种程序设计语言编写程序代码,并最终得到结果 的过程. 为了使计算机能够理解人的意图,人类就必须要将需解决的问题的思路.方法.和手段通 过计算机能够理解的形式告诉计算机,使得计算机能够根据人的指令一步一步去工作,完 成某种特定的任务.这种人和计算机之间交流的过程就是编程.   2.Java 语言概述,历史.特点 是 SUN(Stanford Universit

推荐给新手学习seo基础知识的十本SEO书籍

从2011年起,网络上盛起多家seo优化培训机构,纷纷大包大揽的设堂招生.培训的课程多为一个月时间,学生学习的是被称为"实战"的口语化教程.主要是教大家如何去操作网站SEO优化,如怎么写网站的标签,网站如何布局,在哪里做外链等等.教出来的学生很多都是"只会做,不会想",遇到一些与seo优化培训中不一样的状况就不知道怎么去处理了. seo是一门学问,它是针对于搜索引擎的一门学问.既然是一门学问,想弄懂它就必须从基础开始学习,学习最简单的seo基础知识,慢慢的去深入,最

第二课笔记:搜索引擎基础知识和工作原理

  大家好,我是专门从事SEO的,几个月来一直都在维护和优化按摩器排行榜www.yziyuan.com这个网站,并从中总结了很多的经验和知识.今天要分享的是<搜索引擎基础知识和工作原理>,这是最基本的概念吧, 第一部分:什么是搜索引擎? 1,定义? 官方定义: 搜索引擎是指根据一定的策略.运用特定的计算机程序从互联网上搜集信息,在对信息进行组织和处理后,为用户提供检索服务,将用户检索相关的信息展示给用户的系统.百度和谷歌等是搜索引擎的代表. 我的理解是: 按照搜索引擎的搜索规则去设置目标网站的

基础知识是seo的灵魂

Seoer每天看到seo这三个字,甚至有点审美疲劳了,其实我们在做seo的同时也是在学习seo.我相信在很多人身上会存在这样一问题,做着做着,感觉就迷茫了,不知往哪方面学习,有时感觉又很复杂,遇见问题找相关书籍一一查找.以上问题告诉我们,不懂点最基本的知识是很浪费时间的,越简单的知识越不要忽略,为了给后期工作做铺垫,我们得掌握好基础知识! 相当一部分个人网站对搜索引擎的依赖性太强,使得流量不高.网站排名在一个稳定的范围.ip为恒值是有点难度的,如果想获得更高的用户访问量,站长必须有更扎实的基础知

Groovy使Spring更出色,第1部分: 集成的基础知识

看看 Groovy 如何增加基于 Spring 的应用程序的灵活性 简介:Spring Framework 为 Web 和企业应用程序提供了坚实的基础.通过支 持 Groovy 等动态语言 ,Spring 添加了一些功能,从而使应用程序架构更加灵活.更具动态性.在包含 2 部分的系列文章 的 第一部分中,您将学习将 Groovy 集成到 Spring 应用程序的基础知识. Spring 2.0 支持将动态语言集成到基于 Spring 的应用程序中.Spring 开箱 即用地支持 Groovy.

多核程序设计的相关基础知识----以误差扩散算法为例

    本文从基础入手,主要阐述基于桌面电脑的多核程序设计的基础知识,包括一些向量化运算,虚拟机算,多线程等的相关知识总结. 一.计算平台的分类 单指令单数据流机器(SISD) 传统的串行计算机,所有指令都是串行. 多指令单数据流机器(MISD) 多个指令流同时对一个数据流进行处理.但是大多数情况下,多个指令流处理多个数据才是更加有效的处理方式. 单指令多数据流机器(SIMD) 几乎所有的计算机都实现了SIMD功能,intel处理器中实现的MMX,SSE,SSE2,SSE3等扩展指令集 说到这里

【J2ME3D系列学习文章之一】J2ME3D开发技术和基础知识

本站文章均为 李华明Himi 原创,转载务必在明显处注明: 转载自[黑米GameDev街区] 原文链接: http://www.himigame.com/j2me-kjava/249.html         由于J2me 3D(jsr184)的文章网上也有了不少,这里就不多说了,只是概括的说下针对于j2me 3D开发基本需要了解的知识以及简单阐述几个术语;  第一:开发模式  :         J2me 3D(jsr184)api 分为两种开发模式,一个是立即模式,另一个就是保留模式.这里简