Java高手的30k之路|面试宝典|精通JVM(二)
JVM基本结构
- 类加载子系统:负责将.class文件加载到内存中,并进行验证、准备、解析和初始化。
- 运行时数据区:包括堆(Heap)、方法区(Method Area)、Java栈(Java Stack)、本地方法栈(Native Method Stack)和程序计数器(Program Counter Register)。
- 执行引擎:包括解释器(Interpreter)、即时编译器(JIT Compiler)和垃圾收集器(Garbage Collector)。
- 本地接口:JNI(Java Native Interface)和本地库接口。
上图展示了JVM的基本结构,各组件的布局和相对位置如下:
- Class Loader Subsystem(类加载子系统):负责将字节码文件加载到内存中。
- Execution Engine(执行引擎):包括解释器、即时编译器和垃圾收集器,负责执行字节码。
- Heap(堆):用于存储对象实例和数组,由垃圾收集器管理。
- Method Area(方法区):存储已加载的类信息、常量池、静态变量和JIT编译后的代码。
- Java Stack(Java栈):每个线程都有自己的Java栈,用于存储局部变量、操作数栈、帧数据等。
- Native Method Stack(本地方法栈):用于本地方法的执行。
- Program Counter Register(程序计数器):记录当前线程所执行的字节码的地址。
- JNI(本地接口):Java Native Interface,用于Java与本地代码的交互。
执行引擎
解释器和即时编译器(JIT Compiler)是JVM中执行引擎的重要组成部分,它们的主要职能如下:
解释器(Interpreter)
解释器的主要职能是逐行解释字节码并执行。具体来说:
-
逐行解释执行:解释器将Java字节码逐行解释为机器指令并执行。这种方式的优点是启动快,能够快速响应并执行代码,尤其是对小型或简单程序而言。
-
执行慢:由于解释器需要逐行解释字节码,每次执行都要进行解释,因而执行速度较慢。对于频繁执行的代码段,这种开销会变得很明显。
-
主要用途:解释器在程序启动阶段发挥重要作用,使程序能迅速启动并开始执行。在此过程中,它还可以收集程序的运行数据,帮助JIT编译器进行优化。
即时编译器(JIT Compiler)
JIT编译器的主要职能是将热点代码(经常执行的代码)编译成本地机器码,从而提高执行效率。具体来说:
-
热点代码编译:JIT编译器通过监控程序的运行,识别出频繁执行的代码段(热点代码),并将这些代码编译成机器码,以便直接在CPU上执行。这样可以显著提高执行速度。
-
优化代码:JIT编译器在编译过程中可以进行多种优化,如方法内联(将频繁调用的小方法直接嵌入到调用点)、逃逸分析(确定对象是否可以分配在栈上而不是堆上)、循环展开和消除冗余代码等。通过这些优化,编译后的机器码性能更高。
-
混合模式执行:JVM采用解释器和JIT编译器相结合的混合模式。程序启动时,解释器迅速执行代码并收集运行数据。随着程序的执行,JIT编译器逐步将热点代码编译为高效的机器码,以提高整体执行性能。
-
编译时间和性能权衡:虽然JIT编译会引入一定的编译时间开销,但这个开销通常被编译后执行的性能提升所弥补。JIT编译器在运行时做出的优化决策,使得程序能在不同的运行环境中表现出更好的性能。
本地方法栈(Native Method Stack)是JVM中的一个运行时数据区,用于支持本地方法的执行。本地方法通常是使用本地编程语言(如C、C++)编写的,用来实现Java中无法直接实现的底层操作或者与操作系统交互。具体来说,本地方法栈存放以下内容:
本地方法栈
本地方法栈存储与本地方法调用相关的信息,包括方法的参数、局部变量和返回值等。它类似于Java栈,但专门用于本地方法的执行。
1. 本地方法接口(JNI)的数据
JNI(Java Native Interface)是Java与本地语言交互的接口。本地方法栈包含JNI所需的数据结构,用于管理和执行本地方法调用。这些数据结构包括:
- JNI环境指针:用于访问JNI环境和调用JNI函数。
- JNI局部引用:本地方法中创建的Java对象引用,它们的生命周期在方法执行期间。
3. 操作系统调用栈
当本地方法调用涉及操作系统层面的功能时,本地方法栈也会存储相应的调用栈信息。例如,调用系统API、进行文件I/O操作或网络操作时,操作系统的调用栈信息也会被存放在本地方法栈中。
4. 原生库的信息
本地方法栈可能包含加载的原生库的相关信息。通过JNI加载的本地库(如.dll
或.so
文件)的加载地址和加载状态信息都可能存放在本地方法栈中。
示例
以下是一个简单的示例,展示了Java代码如何调用本地方法:
public class NativeExample {// 声明本地方法public native void printHello();static {// 加载本地库System.loadLibrary("NativeExample");}public static void main(String[] args) {new NativeExample().printHello();}
}
对应的C代码实现如下:
#include <jni.h>
#include <stdio.h>
#include "NativeExample.h"JNIEXPORT void JNICALL Java_NativeExample_printHello(JNIEnv *env, jobject obj) {printf("Hello from native code!\n");
}
在上述示例中,printHello
方法的调用涉及以下步骤:
- Java代码通过JNI调用本地方法。
- JVM将调用信息(参数、局部变量等)推入本地方法栈。
- 本地方法栈中的JNI环境指针和局部引用用于管理和执行本地方法。
- 本地方法执行并可能调用操作系统API,操作系统调用栈信息也存储在本地方法栈中。
类加载器
在Java高级开发中,深入理解类加载器(ClassLoader)是非常重要的。类加载器负责将类文件加载到JVM中,并按照特定的机制进行管理。以下是对启动类加载器、扩展类加载器和应用类加载器的详细介绍。
1. 启动类加载器(Bootstrap ClassLoader)
- 作用:负责加载Java核心类库中的类,如
java.lang.*
、java.util.*
等。 - 实现:启动类加载器是由本地代码实现的(通常是C/C++),并且它是JVM的一部分。
- 类路径:从
<JAVA_HOME>/lib
目录加载类,如rt.jar
、charsets.jar
等。 - 特点:
- 启动类加载器是没有父类加载器的根类加载器。
- 它不是由Java语言实现的,因此在Java程序中无法直接获取启动类加载器的引用。
- 加载系统级的、与平台相关的Java类。
2. 扩展类加载器(Extension ClassLoader)
- 作用:负责加载Java扩展库中的类。
- 实现:扩展类加载器由
sun.misc.Launcher$ExtClassLoader
类实现。 - 类路径:从
<JAVA_HOME>/lib/ext
目录加载类或者从由系统属性java.ext.dirs
指定的目录加载类。 - 特点:
- 扩展类加载器的父类加载器是启动类加载器。
- 它加载的是位于扩展目录中的库,这些库通常提供额外的功能或服务。
3. 应用类加载器(Application ClassLoader)
- 作用:负责加载应用程序类路径(classpath)下的类。
- 实现:应用类加载器由
sun.misc.Launcher$AppClassLoader
类实现。 - 类路径:从系统属性
java.class.path
指定的目录或JAR文件中加载类,即通常的CLASSPATH
环境变量。 - 特点:
- 应用类加载器的父类加载器是扩展类加载器。
- 它加载应用程序自定义的类和库,是开发者最常接触到的类加载器。
- 可以通过
ClassLoader.getSystemClassLoader()
方法获取应用类加载器的实例。
双亲委派模型
Java的类加载机制采用了双亲委派模型。此模型的核心思想是:类加载器在加载一个类时,首先将请求委派给父类加载器,只有当父类加载器无法完成加载任务时,才尝试自己加载。
双亲委派模型的工作流程如下:
- 当一个类加载器收到类加载请求时,它不会自己先去加载,而是将该请求委派给父类加载器。
- 父类加载器接着将该请求委派给它的父类加载器,依此递归下去,直到请求被委派到启动类加载器。
- 启动类加载器尝试加载,如果成功则返回加载结果;如果失败(如类不存在),则将控制权交回给子类加载器。
- 子类加载器接到失败通知后,自己尝试加载该类。
优点:
- 避免重复加载:确保Java核心类库不会被重复加载,确保全局唯一。
- 安全性:防止核心类库被自定义类覆盖。
自定义类加载器
有时我们需要自定义类加载器以满足特定需求,如加载网络上的类、从数据库加载类等。自定义类加载器可以通过继承ClassLoader
类实现,并重写findClass
方法。例如:
public class CustomClassLoader extends ClassLoader {@Overrideprotected Class<?> findClass(String name) throws ClassNotFoundException {byte[] classData = loadClassData(name); // 读取类文件的字节码if (classData == null) {throw new ClassNotFoundException();}return defineClass(name, classData, 0, classData.length);}private byte[] loadClassData(String name) {// 自定义类加载逻辑,例如从网络或数据库加载类字节码return null;}
}
相关文章:

Java高手的30k之路|面试宝典|精通JVM(二)
JVM基本结构 类加载子系统:负责将.class文件加载到内存中,并进行验证、准备、解析和初始化。运行时数据区:包括堆(Heap)、方法区(Method Area)、Java栈(Java Stack)、本…...

JVM专题六:JVM的内存模型
前面我们通过Java是如何编译、JVM的类加载机制、JVM类加载器与双亲委派机制等内容了解到了如何从我们编写的一个.Java 文件最终加载到JVM里的,今天我们就来剖析一下这个Java的‘中介平台’JVM里面到底长成啥样。 JVM的内存区域划分 Java虚拟机(JVM&…...

学习java第一百零七天
解释JDBC抽象和DAO模块 使用JDBC抽象和DAO模块,我们可以确保保持数据库代码的整洁和简单,并避免数据库资源关闭而导致的问题。它在多个数据库服务器给出的异常之上提供了一层统一的异常。它还利用Spring的AOP模块为Spring应用程序中的对象提供事务管理服…...

k8s上尝试滚动更新和回滚
滚动更新和回滚 实验目标: 学习如何进行应用的滚动更新和回滚操作。 实验步骤: 创建一个 Deployment。更新 Deployment 的镜像版本,观察滚动更新过程。回滚到之前的版本,验证回滚操作。 今天呢,我们继续来进行我们k…...

GitHub Copilot 登录账号激活,已经在IntellJ IDEA使用
GitHub Copilot 想必大家都是熟悉的,一款AI代码辅助神器,相信对编程界的诸位并不陌生。 今日特此分享一项便捷的工具,助您轻松激活GitHub Copilot,尽享智能编码之便利! GitHub Copilot 是由 GitHub 和 OpenAI 共同开…...

进程知识点(二)
文章目录 一、进程关系?二、孤儿态进程(Orphan)定义危害处理 三、僵尸进程定义处理 四、守护进程(Daemon )定义作用 总结 一、进程关系? 亲缘关系:亲缘关系主要体现于父子进程,子进程父进程创建,代码继承于父进程&…...

【线性代数】【一】1.6 矩阵的可逆性与线性方程组的解
文章目录 前言一、求解逆矩阵二、线性方程组的解的存在性总结 前言 前文我们引入了逆矩阵的概念,紧接着我们就需要讨论一个矩阵逆的存在性以及如何求解这个逆矩阵。最后再回归上最初的线性方程组的解,分析其中的联系。 一、求解逆矩阵 我们先回想一下在…...

基于大型语言模型的全双工语音对话方案
摘要解读 我们提出了一种能够以全双工方式运行的生成性对话系统,实现了无缝互动。该系统基于一个精心调整的大型语言模型(LLM),使其能够感知模块、运动功能模块以及一个具有两种状态(称为神经有限状态机,n…...

Spring Boot集成Minio插件快速入门
1 Minio介绍 MinIO 是一个基于 Apache License v2.0 开源协议的对象存储服务。它兼容亚马逊 S3 云存储服务接口,非常适合于存储大容量非结构化的数据,例如图片、视频、日志文件、备份数据和容器/虚拟机镜像等,而一个对象文件可以是任意大小&…...

【C++新特性】右值引用
右值和右值的区别 C11 中右值可以分为两种:一个是将亡值( xvalue, expiring value),另一个则是纯右值( prvalue, PureRvalue): 纯右值:非引用返回的临时变量、运算表达式产生的临时变…...

信息安全基础知识(完整)
信息安全基础知识 安全策略表达模型是一种对安全需求与安全策略的抽象概念表达,一般分为自主访问控制模型(HRU)和强制访问控制模型(BLP、Biba)IDS基本原理是通过分析网络行为(访问方式、访问量、与历史访问…...

QT
#include "widget.h" #include "ui_widget.h" Widget::Widget(QWidget *parent) : QWidget(parent) , ui(new Ui::Widget) ,Gcancle(new QPushButton("取消",this)) ,EmmEdit(new QLineEdit(this)) { ui->setupUi(this);…...

双例集合(三)——双例集合的实现类之TreeMap容器类
Map接口有两个实现类,一个是HashMap容器类,另一个是TreeMap容器类。TreeMap容器类的使用在API上于HashMap容器类没有太大的区别。它们的区别主要体现在两个方面,一个是底层实现方式上,HashMap是基于Hash算法来实现的吗,…...

[SAP ABAP] 运算符
1.算数运算符 算术运算符描述加法-减法*乘法/除法MOD取余 示例1 输出结果: 输出结果: 2.比较运算符 比较运算符描述示例 等于 A B A EQ B <> 不等于 A <> B A NE B >大于 A > B A GT B <小于 A < B A LT B >大于或等于 A > B A GE B <小…...

MSPM0G3507 ——GPIO例程讲解2——simultaneous_interrupts
主函数: #include "ti_msp_dl_config.h"int main(void) {SYSCFG_DL_init();/* Enable Interrupt for both GPIOA and GPIOB ports */NVIC_EnableIRQ(GPIO_SWITCHES_GPIOA_INT_IRQN); //启用SWITCHES——A的中断 NVIC_EnableIRQ(GPIO_S…...

某程序员:30岁了,老婆管钱,背着我买了50万股票,亏了20w,强制她清仓后又买了36万
“辛辛苦苦攒了几年钱,本想买房买车,结果全被老婆炒股亏掉了!” 近日,一位30岁的程序员大哥在网上吐苦水,引发了网友们的热议。 这位程序员大哥和妻子结婚后,一直秉持着“男主外,女主内”的传统…...

Docker常见面试题整理
文章目录 1. Docker 是什么?它解决了什么问题?2. Docker 和虚拟机(VM)的区别是什么?3、Docker三个核心概念4、如何构建一个 Docker 镜像?5、如何将一个 Docker 容器连接到多个网络?6、Docker Co…...

35 - 最后一个能进入巴士的人(高频 SQL 50 题基础版)
35 - 最后一个能进入巴士的人 -- sum(weight) over(order by turn) as total,根据turn升序,再求前面数的和 selectperson_name from(selectperson_name,sum(weight) over(order by turn) as totalfromQueue) new_Queue wheretotal<1000 order by total desc lim…...

WPF将dll文件嵌入到exe文件中
WPF将dll文件嵌入到exe文件中 第一步:打开.csproj文件,在Import节点后添加如下代码: <Target Name"AfterResolveReferences"><ItemGroup><EmbeddedResource Include"(ReferenceCopyLocalPaths)" Condit…...

2024年AI+游戏赛道的公司和工具归类总结
随着人工智能技术的飞速发展,AI在游戏开发领域的应用越来越广泛。以下是对2024年AI+游戏赛道的公司和工具的归类总结,涵盖了从角色和场景设计到音频制作,再到动作捕捉和动画生成等多个方面。 2D与3D创作 2D创作工具:专注于角色和场景的平面设计,提供AI辅助的图案生成和风…...

svm和决策树基本知识以及模型评价以及模型保存
svm和决策树基本知识以及模型评价以及模型保存 文章目录 一、SVM1.1,常用属性函数 二、决策树2.1,常用属性函数2.2,决策树可视化2.3,决策树解释 3,模型评价3.1,方面一(评价指标)3.2&…...

C++ 79 之 自己写异常类
#include <iostream> #include <string> using namespace std;class MyOutOfRange : public exception{ // 选中exception右键 转到定义 复制一份 virtual const char* what() const _GLIBCXX_TXN_SAFE_DYN _GLIBCXX_NOTHROW 进行函数重写 public: string m_msg;M…...

如何搭建一个成功的短剧制作平台
要搭建一个成功的短剧制作平台,需要考虑多个方面,包括目标定位、技术选择、内容管理、用户体验等。 1、明确目标和定位: 确定你的目标受众是谁,他们的年龄、兴趣、消费习惯等。 明确短剧制作平台的主要定位,是提供原创…...

kotlin类
一、定义 1、kotlin中使用关键字class 声明类,如果一个类没有类体,也可以省略花括号, 默认为public 类型的: // 这段代码定义了一个公开的、不可被继承的Test类 class Test{} // 没有类体,可以省略花括号 class Test 底层代码&…...

android | studio的UI布局和代码调试 | UI调试 (用于找到项目源码)
网上找到一个项目,想快速的搞懂是怎么实现的,搞了半天发现原来android都升级到Jetpack Compose了,然后去找源码挺不容易的,摸索中发现了这个调试的方法,还可以。 https://developer.android.com/studio/debug/layout-i…...

LangChain实战技巧之六:一起玩转config(上篇)——ConfigurableField
简介 Config 包含两大类内容, ConfigurableField 可配置的字段 configurable_alternatives 可配置的替代方案 分别使用两篇文章来给大家介绍,本篇先介绍ConfigurableField 常规介绍 一些资料会这样介绍 model_spec model.configurable_fields(model…...

扫码称重上位机
目录 一 设计原型 二 后台代码 一 设计原型 模拟工具: 二 后台代码 主程序: using System.IO.Ports; using System.Net; using System.Net.Sockets; using System.Text;namespace 扫码称重上位机 {public partial class Form1 : Form{public Form1(){Initialize…...

操作系统入门 -- 进程的通信方式
操作系统入门 – 进程的通信方式 1.什么是进程通信 1.1 定义 进程通信就是在不同进程之间交换信息。在之前文章中可以了解到,进程之间相互独立,一般不可能互相访问。因此进程之间若需要通信,则需要一个所有进程都认可的共享空间࿰…...

Python读取wps中的DISPIMG图片格式
需求: 读出excel的图片内容,这放在微软三件套是很容易的,但是由于wps的固有格式,会出现奇怪的问题,只能读出:类似于 DISPIMG(“ID_2B83F9717AE1XXXX920xxxx644C80DB1”,1) 【该DISPIMG函数只有wps才拥有】 …...

elasticsearch的入门与实践
Elasticsearch是一个基于Lucene构建的开源搜索引擎。它提供了一个分布式、多租户能力的全文搜索引擎,具有HTTP web接口和无模式的JSON文档。以下是Elasticsearch的入门与实践的基本步骤: 入门 安装Elasticsearch: 从Elasticsearch官网下载对…...