Java核心:类加载和JVM内存的分配

类的加载:

  指的是将class文件的二进制数据读入到运行时数据区(JVM在内存中划分的)

中,并在方法区内创建一个class对象。

类加载器:

  负责加载编译后的class文件(字节码文件)到JVM(Java虚拟机)当中。

而类加载器主要分为以下几种:

1.Bootstrap class loader (引导类加载器)

  负责加载Java核心类库。在jre\lib目录下,包括rt.jar(Java基础类库),这些

都是Java的核心类库。而且这个加载器是由C语言编写的,所以在Java程序中是获取

不到的。

2.Extension class loader(扩展类加载器)

  负责加载Java平台下扩展功能的jar包,这些jar包在jre\lib\ext目录下。这个加载

器由Java语言编写的。

3.System class loader(系统类加载器)

  负责加载classpath目录下的所有类库,classpath目录下的class文件一般

是我们自己写的java文件编译后的。而这个加载器是由Java语言写的。

这些类加载器协同起来完成整个类的加载过程,因此这些类的加载模式基于

”双亲委托模型“。

双亲委托模型“:

  程序运行后,编译器把Java文件编译成class文件后,首先负责加载的是

系统类加载器,但它不会马上加载,而是将此任务移送给它的父类加载器扩展

类加载器加载,扩展类加载器也是将此任务移送给引导类加载器加载。

  class文件到了引导类加载器那,它先判断能不能加载这个类,如果能,就

加载;不能,移送给其子加载器,以此类推。最终我们编写的class都会配置在

classpath环境中,所以,这个类加载任务还是由系统类加载器完成。如果系统

类加载器都不能加载,就抛出ClassNotFoundException。

当一个class加载到JVM中,类加载阶段已经完成。接下来JVM分配内存,对整个

class文件(文件里面都是二进制的汇编命令)进行内容解析(JVM对二进制的命

令逐行解析,交由CPU执行)。

内存分配:

  JVM运行起来时就给内存划分空间,这块空间就称为运行时数据区。

运行时数据区被划分为以下几块内容:

1.栈:

  每一个线程运行起来的时候就会对应一个栈(线程栈),栈中存放的数据被当前

线程所独享(不会产生资源共享情况,所以线程是安全的)。而栈当中存放的是栈帧,

当线程调用方法时,就是形成一个栈帧,并将这个栈帧进行压栈操作。方法执行完后,

进行出栈操作。这个栈帧里面包括(局部变量,操作数栈,指向当前方法对应类的常

量池引用,方法返回地址等信息)。

2.本地方法栈:

  本地方法栈的机制和栈的相似,区别在于,栈运行的是Java实现的方法,而本地

方法栈运行的是本地方法。本地方法指的是JVM需要调用非Java语言所实现的方法,

例如C语言。在JVM规范中,没有强化性要求实现方一定要划分出本地方法栈(例如:

HotSpot虚拟机将本地方法栈和栈合二为一)和具体实现(不同的操作系统,对JVM

规范的具体实现都不一样)。

3.程序计数器:

  程序计数器也可以称为PC寄存器(通俗讲就是  指令缓存)。它主要用于缓存当前

程序下一条指令的指令地址,CPU根据这个地址找到将要执行的指令。这个寄存器是JVM

内部实现的,不是物理概念上的计数器,不过和JVM的实现逻辑一样。

4.堆:

  堆内存主要存放创建的对象和数组。堆内存在JVM中是唯一的,能被多个线程所共享。

堆里面的每一个对象都存放着实例的实例变量。

  当在方法中定义了局部变量,如果这个变量是基本数据类型,那么这个变量的值就直接

存放在栈中;如果这个变量是引用数据类型,那么变量值就存放在堆内存中,而栈中存放的是

指向堆中的引用地址。

5.方法区:

  方法区在JVM也是一个非常重要的一块内存区域,和堆一样,可以被多个线程多共享。

主要存放每一个加载class的信息。class信息主要包含魔数(确定是否是一个class文件),常量

池,访问标志(当前的类是普通类还是接口,是否是抽象类,是否被public修饰,是否使用了final

修饰等描述信息......),字段表集合信息(使用什么访问修饰符,是实例变量还是静态变量,是否

使用了final修饰等描述信息.....),方法表集合信息(使用什么访问修饰符,是否静态方法,是否

使用了final修饰,是否使用了synchronized修饰,是否是native方法......)等内容。当一个类加

载器加载了一个类的时候,会根据这个class文件创建一个class对象,class对象就包含了上述的信息。

后续要创建这个类的实例,都根据这个class对象创建出来的。

6.常量池:

  常量池是方法区中的一部分,存放class对象中最重要的资源。JVM为每一个class对象都维护一个

常量池。它主要存储两种类型的常量:

1.字面常量:

  字面常量通常就是在Java中定义的字面量值。例如:int = 1,中的 1,String s = "hello",这个

hello就是字面量。或者使用final修饰的常量值。

2.符号引用:

  符号引用主要包括类和接口的完整类名,属性的名称和描述符,方法名和描述符等。

-----------------------------------------------------------

时间: 2024-10-13 11:13:57

Java核心:类加载和JVM内存的分配的相关文章

ClassLoader类加载机制&&JVM内存管理

一.ClassLoader类加载机制 在java中类加载是遵循委派双亲加载的:通过调用loadClass方法逐级往上传递委派加载请求,当找不到父ClassLoader时调用其findClass方法尝试进行查找和加载,如果当前ClassLo找不所需的Class,则由其孩子尝试进行查找和加载,如果当前ClassLoader找了所需的Class则将该Class按请求路径逐级返回孩子.其关系图如下所示: ClassLoader.loadClass(...) 是ClassLoader的入口点.当一个类没有

Java学习之:JVM内存模型

一.文章来由 开始实习啦,实习转战Java开发工程师... 二.JVM内存模型总图 Java中通过多线程机制使得多个任务同时执行处理,所有的线程共享JVM内存区域main memory,而每个线程又单独的有自己的工作内存,当线程与内存区域进行交互时,数据从主存拷贝到工作内存,进而交由线程处理(操作码+操作数). 在之前,我们也已经提到,JVM的逻辑内存模型如下: 三.JVM内存模型详解 1.程序计数器 程序计数器(Program Counter Register)是一块较小的内存空间,它的作用可

java中OutofMemoryError和JVM内存结构

OutOfMemoryError在开发过程中是司空见惯的,遇到这个错误,新手程序员都知道从两个方面入手来解决: 1:是排查程序是否有BUG导致内存泄漏: 2:是调整JVM启动参数增大内存. OutOfMemoryError有好几种情况,每次遇到这个错误时,观察OutOfMemoryError后面的提示信息,就可以发现不同之处,如: 引用 java.lang.OutOfMemoryError: Java heap space java.lang.OutOfMemoryError: unable t

Java性能剖析]Sun JVM内存管理和垃圾回收

内存管理和垃圾回收是JVM非常关键的点,对Java性能的剖析而言,了解内存管理和垃圾回收的基本策略非常重要.本篇对Sun JVM 6.0的内存管理和垃圾回收做大概的描述. 1.内存管理      在程序运行过程当中,会创建大量的对象,这些对象,大部分是短周期的对象,小部分是长周期的对象,对于短周期的对象,需要频繁地进行垃圾回收以保证无用对象尽早被释放掉,对于长周期对象,则不需要频率垃圾回收以确保无谓地垃圾扫描检测.为解决这种矛盾,Sun JVM的内存管理采用分代的策略.      1)年轻代(Y

java复习基础篇—-JVM内存结构(转)

主要内容如下: JVM启动流程 JVM基本结构 内存模型 编译和解释运行的概念 一.JVM启动流程: JVM启动时,是由java命令/javaw命令来启动的. 二.JVM基本结构: JVM基本结构图: <深入理解Java虚拟机(第二版)>中的描述是下面这个样子的: Java中的内存分配: Java程序在运行时,需要在内存中的分配空间.为了提高运算效率,就对数据进行了不同空间的划分,因为每一片区域都有特定的处理数据方式和内存管理方式. 具体划分为如下5个内存空间:(非常重要) 栈:存放局部变量

java 命令简单查看jvm内存使用状况

1.jps 虚拟机进程状况工具 语法  jsp [option] [hostid] 选项 作用 -q 只显示lvmid -m 输出虚拟机启用时传入main方法参数 -l 输出主类全名 -v 输出jvm参数 注意: 如果在linux环境中jps命令不能输出vmid数据(jdk1.6.0.23/24 bug),需要修改catalina.sh文件 if [ -z "$CATALINA_TMPDIR" ] ; then?# Define the java.io.tmpdir to use fo

JVM内存分配及String常用方法

一,JVM内存分配和常量池 ? 在介绍String类之前,先来简单分析一下在JVM中,对内存的使用是如何进行分配的.如下图所示(注意:在jdk1.8之后便没有方法区了): ? ? 如上JVM将内存分为多个不同的区域,这些区域都有各自的用途.创建和销毁的时间,有些区域随虚拟机进程的启动而存在,有些区域则是依赖用户线程的启动和结束来建立和销毁. ? 区域名称的说明: 1.1,方法区: ? 属于数据共享内存区域,存储已被虚拟机加载的类信息.常量.静态变量.即时编译器编译后的代码等数据. 1.2,虚拟机

JAVA学习总结之JVM概述

参考博客 JVM理解其实并不难! JVM的内存区域划分 Java垃圾回收机制 JVM内存区域 由于Java程序是交由JVM执行的,所以我们在谈Java内存区域划分的时候事实上是指JVM内存区域划分.在讨论JVM内存区域划分之前,先来看一下Java程序具体执行的过程: 如上图所示,首先Java源代码文件(.java后缀)会被Java编译器编译为字节码文件(.class后缀),然后由JVM中的类加载器加载各个类的字节码文件,加载完毕之后,交由JVM执行引擎执行.在整个程序执行过程中,JVM会用一段空

JVM内存模型详解

JVM内存模型也叫JVM运行时区域,是认识和了解JVM工作原理的基础,从java诞生以来,JVM内存模型基本保持着大同小异的整体形态,由此可见JVM内存模型是相当稳定的,直到jdk1.8之后JVM内存模型中才将permGen(永生代),也就是过去的方法区完全去除,使用metaspace取而代之,但是从整个JVM内存形态来说其实并没有产生太大的变化,有点"换汤不换药"的味道.除此之外JVM内存模型的设计原理也充分考虑了java程序的运行过程以及GC的策略,所以JVM内存模型是一个既基础又