JVM快速入门(1)JVM体系结构、运行时数据区、类加载器、线程共享和独享、分区、Java对象实例化

news2024/11/18 16:26:08

5.1 JVM体系结构

在这里插入图片描述

  • 线程独占区-程序计数器(Program Counter Register)

    • 程序计数器是一块较小的内存空间,它可以看做是当前线程所执行的字节码的行号指示器;
    • 在虚拟机的概念模型里,字节码解释器工作时就是通过改变这个计数器的值来选取下一条需要执行的字节码指令,分支、循环、跳转、异常处理、线程恢复等基础功能都需要依赖这个计数器来完成;
    • 如果线程正在执行的是一个Java方法,这个计数器记录的是正在执行的虚拟机字节码指令的地址;如果正在执行的是Native方法,这个计数器值则为空(Undefined)。此内存区域是唯一一个在Java虚拟机规范中没有规定任何OutOfMemoryError情况的区域。
  • 线程独占区-Java虚拟机栈(Java Virtual machine Stacks)

    • Java虚拟机栈与线程生命周期相同。其描述的是Java方法执行的内存模型:每个方法在执行的同时都会创建一个栈帧(Stack Frame)用于存储局部变量表、操作数栈、动态链接、方法出口等信息。每一个方法从调用直至执行完成的过程,就对应着一个栈帧在虚拟机中从入栈到出栈的过程。

    • 局部变量表存放了编译器可知的各种基本数据类型(boolen,byte,char,short,int,float,long,double),对象引用(reference类型)和returnAddress类型(指向了一条字节码指令的地址)。

    • 在这个区域中,Java虚拟机规范规定了两种异常情况:

    • 如果线程请求的栈深度大于虚拟机所允许的深度,抛出StackOverFlowError异常;
    • 如果虚拟机栈可以动态扩展(当前大部分Java虚拟机都可动态扩展,只不过Java虚拟机规范中也允许固定长度的虚拟机栈),并且扩展时无法申请到足够的内存,就会抛出OutOfMemoryError异常。
  • 线程独占区-本地方法栈(Native Method Stack)

    • 本地方法栈与虚拟机栈所发挥的作用非常相似,区别是:虚拟机栈为虚拟机执行Java方法(也就是字节码)服务,本地方法栈为虚拟机使用到的Native方法服务。
    • 由于虚拟机规范中没有对本地方法栈中的语言、使用方式与数据结构进行强制规定,有的虚拟机(入Sun HotSpot虚拟机)直接把本地方法栈和虚拟机栈合二为一。
    • 本地方法栈会抛出StackOverflowError异常和OutOfMemoryError异常。
  • 线程共享区-Java堆(Java Heap)

    • Java堆是Java虚拟机所管理的内存中最大的一块,在虚拟机启动时创建,其唯一目的就是存放对象实例:所有的对象实例以及数组都要在堆上分配(但随着JIT编译器的发展与逃逸分析技术逐渐成熟,所有的对象分配在堆上也渐渐不是那么绝对了)。
    • Java堆是垃圾收集器管理的主要区域,现在收集器基本采用分代收集算法,所以Java堆还可细分为:新生代和老年代;再细致点分为Eden空间(伊甸园区),From Survivor空间(幸存区0),To Survivor空间(幸存区1)等;
    • 根据Java虚拟机规范的规定,Java堆可以处于物理上不连续的内存空间中,只要逻辑上连续即可;
    • 如果堆中没有内存完成实例分配,并且堆也无法再扩展时,将会抛出OutOfMemoryError异常。
  • 线程共享区-方法区(Method Area)

    • 方法区用于存储已被虚拟机加载的类信息(构造方法、接口定义)、常量、静态变量、即时编译器编译后的代码(运行时常量池)等数据
    • 所谓的方法区为永久代的说法,仅仅是因为HotSpot虚拟机将GC分代收集扩展至方法区,或者说使用永久代来实现方法而已。对于其他虚拟机是不存在永久代的说法的。
    • 运行时常量池是方法区的一部分,用于存放编译器生成的各种字面量和符号引用,一般还会存放翻译出来的直接引用。这部分内容将在类加载后进入方法区的运行时常量池中存放。当常量池无法再申请到内存时抛出OutOfMemoryError异常。
  • 举例:

            String str1 = "abd";
            String str2 = new String("abd");
    
            System.out.println(str1 == str2);
            System.out.println(str1 == str2.intern());
    
    输出:
        false
        true
    
    分析:
       (1) String str1 = "abc" ,str1指向常量池;
       (2) String str2 = new String("abc");str2指向堆内存对象,二者地址不同所以str1 == str2 结果为false;
       (3) 但是str2.intern()会把字符串值从堆内存移动到常量池中(如果常量池存在则返回该值的地址),这样一来str2和str1都是指向常量池的abc。  
        如下图所示:     
    

    在这里插入图片描述

5.2 JVM详细架构图

在这里插入图片描述

5.3 JVM架构之运行时数据区

  • 线程共享区包括:堆、元空间
  • 线程私有区包括:虚拟机栈、本地方法栈、程序计数器

运行时数据区

包括:程序计数器(PC寄存器)、Java虚拟机栈、Java堆、方法区、运行时常量池、本地方法栈等等。

5.3.1 PC 寄存器,也叫程序计数器
  • 1、JVM支持多个线程同时运行,每个线程拥有一个程序计数器,是线程私有的,用来存储指向下一条指令的地址。
  • 2、在创建线程的时候,创建相应的程序计数器。
  • 3、执行本地native方法时,程序计数器的值为undefined。
  • 4、是一块比较小的内存空间,是唯一一个在JVM规范中没有规定OutOfMemoryError的内存区域。
5.3.2 虚拟机栈
  • 栈是由一系列帧(Frame)组成(因此Java栈也叫作帧栈),是线程私有的。
  • 帧是用来保存一个方法的局部变量、操作数栈(java没有寄存器,所有的参数传递使用操作数栈)、常量池指针、动态链接、方法返回值等。
  • 每一次方法调用创建一个帧并压栈,退出方法的时候,修改栈顶指针就可以把栈帧中的内容销毁。
  • 局部变量表存放了编译期可知的各种基本数据类型和引用数据类型、每个slot存放32位的数据,long、double占两个槽位。
  • 栈的优点:存取速度比堆快,仅次于程序计数器。
  • 栈的缺点:存在栈中的数据太小,生存期是在编译期决定的,缺乏灵活性。
  • StackOverflowError异常:当线程请求的栈深度大于虚拟机所允许的深度;
  • OutOfMemoryError异常:如果栈的扩展时无法申请到足够的内存。
5.3.3 Java堆
  • 用来存放应用系统创建的对象和数组,所有线程共享Java堆。
  • GC主要管理堆空间,对分代GC来说,堆也是分代的。
  • 堆的优点:运行期动态分配内存大小,自动进行垃圾回收。
  • 堆的缺点:效率相对较慢。
5.3.4 方法区的理解
  • 对于HotSpotJVM而言,方法区还有一个别名,叫做Non-Heap(非堆),目的就是和堆区分开,所以方法区看做是一块独立于Java堆的内存空间。
  • 方法区(Method Area)和Java堆一样,是各个线程共享的内存区域;
  • 方法区在JVM启动时被创建,并且它的实际的物理内存空间和Java堆区一样,都是可以不连续的;
  • 方法区的大小,和堆空间一样,可以选择固定大小或者可扩展;
  • 方法区的大小,决定了系统可以保持多少个类,如果系统定义了太多的类,导致方法区溢出,虚拟机同样会抛出内存溢出错误:java.lang.OutOfMemoryError:Metaspace;
  • 关闭JVM就是释放这个区域的内存。

HotSpot中方法区的演进:

  • JDK1.6及之前,有永久代,静态变量存放在永久代上,常量池在方法区了;
  • JDK1.7,有永久代,但已经逐步“去永久代”,字符串常量池、静态变量移除,保存在堆中;
  • JDK1.8及之后,无永久代,类型信息、字段、方法、常量保存在本地内存的元空间。但字符串常量池、静态变量仍在堆上。
  • 永久代、元空间二者并不只是名字改变了,内部结构也调整了。
5.3.5 运行时常量池:
  • 运行时常量池是方法区的一部分;

  • 常量池是Class文件的一部分,用于存放编译期生成的各种字面量和符号引用,这部分内容将在类加载后存放到方法区的运行时常量池中;

  • 在加载类和接口到JVM中,就会创建对应的运行时常量池;

  • JVM为每个已加载的类型(类或接口)都维护一个常量池,池中的数据项像数组一样,通过索引访问的;

  • 运行时常量池中包含各种不同的常量,包括编译期就已经明确的数值字面量,也包括到运行期解析后才能获得的方法或字段引用。此时不再是常量池中的符合地址了,这里换为真实地址:

  • 运行时常量池,相对于Class文件常量池的另一重要特征是:具备动态性;
  • String.intern()
  • 党创建类或接口的运行时常量池时,如果构造运行时常量池所需的内存空间超过了方法区所能提供的最大值,则JVM就会抛出OOM异常。
5.3.6 本地方法栈
  • 在JVM中用来支持native方法执行的栈就是本地方法栈。
  • 在JVM规范中,并没有对本地方发展的具体实现方法以及数据结构作强制规定,虚拟机可以自由实现它。在HotSopt虚拟机中直接就把本地方法栈和Java栈合二为一。

在这里插入图片描述

5.4 类加载器

  • 作用:加载class文件

  • 加载器:

    • 1.虚拟机自带的加载器
    • 2.启动类(根)加载器(Bootstrap classLoader):主要负责加载核心的类库(java.lang.*等),构造ExtClassLoader和AppClassLoader。
    • 3.扩展类加载器(ExtClassLoader):主要负责加载jre/lib/ext目录下的一些扩展的jat.
    • 4.应用程序加载器(AppClassLoader):主要负责加载应用程序的主函数类。
  • 图示:

    在这里插入图片描述

  • 双亲委派机制

    • (1)类加载器收到类加载的请求;

    • (2)将这个请求向上委托给父类加载器去完成,一直向上委托,直到启动类加载器(根加载器);

    • (3)启动加载器检查是否能够加载当前这个类,能加载就结束,使用当前的加载器;否则,抛出异常,通知子加载器进行加载;

    • (4)重复步骤3

    ​ Class Not Found

    null: java 调用不到 c/c++写的调用不到

5.5 Java对象的实例化过程

java世界里面对象无处不在,在创建对象的时候主要经过哪些步骤?

5.5.1 对象的创建过程

类加载检查–>分配内存–>初始化零值–>设置对象头–>执行init方法

如图:

在这里插入图片描述

1.类加载检查
   虚拟机遇到一条new指令时,首先去检查这个指令的参数是否能在常量池中定位到一个类的符号引用并且检查这个符号引用代表的类是否被加载,解析,初始化过,如果没有,那必须执行相应的类加载过程

new 的指令对应到语言层面上讲: new关键词,对象克隆,对象序列化等

2.分配内存
在类加载检查通过后,接下来虚拟机将为新生对象分配内存,对象所需内存的大小在类加载完成后便可完成确定,为对象分配空间的任务等同于把一块确定大小的内存从Java堆中划分出来
3.初始化零值

内存分配完成后,虚拟机需要将分配到的内存空间都初始化为零值()不包括对象头

4.设置对象头(分不同的操作系统,如32位的,64位的)

初始化零值之后,虚拟机要对对象进行必要的设置,例如这个对象是哪个的实例,如何才能找到类的元数据信息,对象的哈希码,对象的GC分代年龄等这些信息,这些信息存放在对象的对象头Object Header之中 在HotSpot虚拟机中,对象在内存中的储存布局可以分为3块区域: 对象头(Header),实例数据(Instance Data)和对齐填充(Padding)

HotSpot虚拟机的对象头包括两部分信息:

     第一部分: 储存对象本的运行时数据,如哈希码,GC分代年龄,锁状态标志,线程持有的锁,偏向线程ID,偏向时间戳等

    第二部分: 类型指针,即对象指向它的类元数据的指针,虚拟机通过这个指针来确定这个对象是哪个类的实例

如下是32 位的对象头

在这里插入图片描述

5.执行init()方法

对象按照程序员的意愿进行初始化;对应到语言层面来讲,就是属性赋值(注意:这是程序员自己赋的值)和执行构造方法

5.5.2 从Java源码到编译class到加载整体过程

对象创建的过程,主要经过如下5步:

  1. 判断类有没有被加载

  2. 如果没有被加载过(才开始加载类(就是类的加载过程))

  3. 初始化 :就是给一些变量进行初始化。

  4. 设置对象头(比较难理解)。

  5. 执行方法: 对对象进行赋值,和执行构造方法。

这里再从源码.java文件到编译的.class文件到加载,详细描述第2步中的类的加载过程:

加载.class文件的时候 window系统下调用底层的应该jvm.dll文件创建java虚拟机去创建一个引导类加载器(C++实现的) 此时java虚拟机已经创建 此时会调用java实现的类加载器启动 加载loadClass方法加载真正的磁盘文件上面的字节码文件,再去发起调用main()方法,此时程序就启动了

类加载到使用整个过程有如下几步: 加载 >> 验证 >> 准备 >> 解析 >> 初始化 >> 使用 >> 卸载

1、加载:

​ 在硬盘上查找并通过IO读入字节码文件,使用到类时才会加载,例如调用 类的main()方法,new对象等等

2、验证:

​ 校验字节码文件的正确性

3、准备:

​ 给类的静态变量分配内存,并赋予默认值 比如Boolean类型默认 false 这些默认值是java虚拟机自己规定的,如果是加final修饰直接就会变成常量 直接赋值了

4、解析:

​ 将符号引用替换为直接引用,就是会将该阶段会把一些静态方法(符号引用,比如 main()方法)替换为指向数据所存内存的指针或句柄等(直接引用),这是所谓的静态链 接过程(类加载期间完成),动态链接是在程序运行期间完成的将符号引用替换为直接 引用,下节课会讲到动态链接(在类加载的时候可能不会加载 只有程序运行到这里才会去加载)

5、初始化:

对类的静态变量初始化为指定的值,执行静态代码块

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1540411.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

如何解决Layui后台接口返回数据,但是table.render不渲染表格数据的问题

我这边进行了pareData数据格式转换,response重新定义了layui的参数格式规范 接口正常返回了数据 但是就是不渲染,我这个郁闷啊!! 忽然,我把后台重新定义的layui规定的格式参数,有个参数名叫data&#xff0…

Day 1 二分算法(C++)

算法简介 二分查找(Binary Search)是一种常见的查找算法,它适用于已经排序好的数组或列表。它的基本思想是不断地将待查找区间分成两半,并通过比较目标值与中间元素的大小关系来确定目标值在哪一半中,从而缩小查找范围…

【leetcode热题】 位1的个数

编写一个函数,输入是一个无符号整数(以二进制串的形式),返回其二进制表达式中数字位数为 1 的个数(也被称为汉明重量)。 提示: 请注意,在某些语言(如 Java)中…

2024.03.23 健身打卡第 34 天

君之英才,实乃盖世无双,渴求统一大业却属鲲鹏之志。然涓滴之水汇成江河,已属不易,奔流向前,汇入大海之时,更会倍感自身之渺茫。 2024.03.23 健身打卡第 34 天

架构整洁之道-读书总结

1 概述 1.1 关于本书 《架构整洁之道》(Clean Architecture: A Craftsman’s Guide to Software Structure and Design)是由著名的软件工程师Robert C. Martin(又称为Uncle Bob)所著。这本书提供了软件开发和架构设计的指导原则…

王道C语言督学营OJ课后习题(课时14)

#include <stdio.h> #include <stdlib.h>typedef char BiElemType; typedef struct BiTNode{BiElemType c;//c 就是书籍上的 datastruct BiTNode *lchild;struct BiTNode *rchild; }BiTNode,*BiTree;//tag 结构体是辅助队列使用的 typedef struct tag{BiTree p;//树…

文件包含一-WEB攻防-PHP应用文件包含LFIRFI伪协议编码算法无文件利用黑白盒

演示案例&#xff1a; 文件包含-原理&分类&利用&修复黑盒利用-VULWEB-有无包含文件白盒利用-CTFSHOW-伪协议玩法 #文件包含-原理&分类&利用&修复 1、原理 程序开发人员通常会把可重复使用的函数写到单个文件中&#xff0c;在使用某些函数时&#xff0c…

MySQL---视图

目录 一、介绍 二、语法 三、视图的更新 四、视图作用 一、介绍 视图&#xff08;View&#xff09;是一种虚拟存在的表。视图中的数据并不在数据库中实际存在&#xff0c;行和列数据来自定义视图的查询中使用的表&#xff0c;并且是在使用视图时动态生成的。 通俗的讲&#…

CSS时钟案例

文章目录 1. 演示效果2. 分析思路3. 代码实现 1. 演示效果 2. 分析思路 背景是表盘&#xff0c;不用自己制作然后用CSS的定位做时针&#xff0c;分针和秒针黑点用伪元素::after生成转动用animation实现 3. 代码实现 <!DOCTYPE html> <html lang"en">&…

密码学之哈希碰撞和生日悖论

哈希碰撞 哈希碰撞是指找到两个不一样的值&#xff0c;它们的哈希值却相同 假设哈希函数的取值空间大小为k &#xff0c;计算次数为n 先算每个值不一样的概率P’ 所以至少两个值相同(即存在哈希碰撞)的概率P为 生日悖论 假设班里有50个人&#xff0c;求班里至少两个人相同…

java.lang.ClassNotFoundException: kotlin.jvm.internal.Intrinsics

今天在使用springBoot连接influxdb报错 java.lang.ClassNotFoundException: kotlin.jvm.internal.Intrinsics 详细报错如下&#xff0c;提出我们缺少一个依赖 原因是由于创建influxdb客户端缺少Kotlin运行时库 解决办法就是 1.显示的添加okhttp的依赖 <dependency>…

如何用VSCode和Clangd与Clang-Format插件高效阅读Linux内核源码及写驱动

一、如何高效阅读Linux源码&#xff1a;基于clangd uboot/busybox等都可以用这种方式&#xff0c;理论上说所有基于Make和Cmake的源码工程都可以用这套方案 阅读Linux源码最大问题在于调用链太复杂&#xff0c;一个函数或变量引用太多&#xff0c;source insight和cscope等基于…

Uibot6.0 (RPA财务机器人师资培训第1天 )RPA+AI、RPA基础语法

训练网站&#xff1a;泓江科技 (lessonplan.cn)https://laiye.lessonplan.cn/list/ec0f5080-e1de-11ee-a1d8-3f479df4d981(本博客中会有部分课程ppt截屏,如有侵权请及请及时与小北我取得联系~&#xff09; 紧接着小北之前的几篇博客&#xff0c;友友们我们即将开展新课的学习~…

Nacos部署(三)Docker部署Nacos2.3单机环境

&#x1f60a; 作者&#xff1a; 一恍过去 &#x1f496; 主页&#xff1a; https://blog.csdn.net/zhuocailing3390 &#x1f38a; 社区&#xff1a; Java技术栈交流 &#x1f389; 主题&#xff1a; Nacos部署&#xff08;三&#xff09;Docker部署Nacos2.3单机环境 ⏱️…

Python数据结构实验 队列的实现

一、实验目的 1&#xff0e;掌握用Python定义队列的顺序存储结构和链式存储结构&#xff0c;以便在实际背景下灵活运用&#xff1b; 2&#xff0e;掌握队列的特点&#xff0c;即先进先出的原则&#xff1b; 3&#xff0e;掌握队列的基本操作实现方法。 二、实验环境 1&…

BufferedInputStream详解

咦咦咦&#xff0c;各位小可爱&#xff0c;我是你们的好伙伴——bug菌&#xff0c;今天又来给大家普及Java之IO流啦&#xff0c;别躲起来啊&#xff0c;听我讲干货还不快点赞&#xff0c;赞多了我就有动力讲得更嗨啦&#xff01;所以呀&#xff0c;养成先点赞后阅读的好习惯&am…

javase day10笔记

第十天课堂笔记 debug调试★★★ 步骤: 设置断点 - > 启动调试debug -> 单步运行 -> 观察参数 单步跳过f8: 向下执行语句,不进入方法内部单步跳入f7: 进入方法内部执行单步跳出shift f8: 跳出当前方法,到方法调用处跳转到光标所在的位置alt f9: 变量整合 变量 …

【力扣hot100】1. 两数之和 49.字母异位词分组 128. 最长连续序列

目录 1. 两数之和题目描述做题思路参考代码 49.字母异位词分组题目描述做题思路参考代码 128. 最长连续序列题目描述做题思路参考代码 1. 两数之和 题目描述 给定一个整数数组 nums 和一个整数目标值 target&#xff0c;请你在该数组中找出 和为目标值 target 的那 两个 整数…

亚马逊云科技《生成式 AI 精英速成计划》

最近亚马逊云科技推出了「生成式AI精英速成计划」&#xff0c;获取包含&#xff1a;免费学习热门生成式AI课程、技能证书、人力主管的面试辅导、云计算国际认证、免费去往北美参加全球用户大会等&#xff5e; 针对开发者和企业非技术专业人士&#xff0c;了解如何使用大模型平台…

【机器学习之---统计】统计学基础概念

every blog every motto: You can do more than you think. https://blog.csdn.net/weixin_39190382?typeblog 0. 前言 统计学基础 1. 频率派 频率学派&#xff08;传统学派&#xff09;认为样本信息来自总体&#xff0c;通过对样本信息的研究可以合理地推断和估计总体信息…