当前位置: 首页 > news >正文

【快速入门】JVM之类加载机制与Native

感慨:
如何定义一个合格的Java程序员,Java程序员要了解掌握哪些知识点,网上的面试题太多了,后端需要了解掌握的知识点太多太多了,Java基础、数据结构、异常、多线程、Spring、Spring boot、事务、算法、数据库(Oracle、MySQL等)、缓存、中间件(各种类型的)、并发异步、消息中间件、微服务、netty(最起码要知道有这个东西吧)、大数据相关(Hive、spark、flink等)、JVM、网络、日志等等等等,太多太多了,每一个单独拿出来都有太多的点了。
所以说,后端Java开发并不是单纯的学习框架+编写业务逻辑,这样简直太简单了,也就是太容易被替代了。
最近问身边很多工作多年的程序员关于jvm相关的知识,基本没有人答得出来,甚至是一些比较浅显的概念点,都说了解这个有什么用呢?实际开发是没有用,但是我还是觉得要深入思考,要考虑底层原理,知其然且知其所以然。
所以一直想写一个关于JVM的文章,不用太复杂但是可以把一些基本的知识概念概括一下,但是jvm的点太多了,我们就说个大概吧,也可以理解为面试知识点向,所以标题是快速入门。

前言

JVM专栏分为以下几个部分:类加载相关、Native相关、PC寄存器(概念)、堆、栈、GC相关

文章目录

  • 前言
  • 类加载
    • 类加载的过程
      • 加载
      • 验证
      • 准备
      • 解析
        • 符号引用与直接引用
      • 初始化
        • init和clinit区别
    • 类加载器
      • 三种类加载器
      • 双亲委派
      • 好处
  • Native
    • 概念
    • JNI

类加载

在Java中,每一个类(.java文件)再通过编译器后,都会形成一个.class文件。
类加载机制指的是将这些.class文件中的二进制数据读入到内存中,并对数据进行校验,解析和初始化。最终,每一个类都会在方法区保存一份它的元数据,在堆中创建一个与之对应的Class对象。

类加载的过程

类的加载过程分 5 个阶段:加载、验证、准备、解析、初始化,其中验证、准备、解析可以归纳为连接阶段。如图示:
在这里插入图片描述
虽然五个阶段之间有箭头指向,但是并不是严格的按照顺序完成,在类加载过程中,这些阶段会交叉混合执行以完成类的加载及初始化。

加载

加载主要是通过类加载器将.class文件读入内存的过程,主要是完成以下操作:

  • 通过类的全限定名定位到.class文件。
  • 读取.class文件的二进制数据,转化为方法区的运行时数据结构(方法区jdk8后由元空间实现,且不在jvm内存中了,而是直接放到了本地内存中,不受jvm参数限制,这个后面具体再讲)。
  • 在JVM堆内存Eden区(伊甸园区)中生成一个java.lang.Class对象,即程序中使用任何类时,系统都会为之建立一个java.lang.Class对象(Class模板。它是用来表示这个类的Class对象!一个类的实例对象可以有多个,但Class对象只有一个,一个类的class对象与其类的元数据一一对应),系统中所有的类都是java.lang.Class的实例。

验证

字面意思很好理解,就是对.class文件进行一系列的校验呗。目的是确保 Class 文件的字节流中包含的信息符合当前虚拟机的要求,并且不会危害虚拟机自身的安全。
包括(具体不展开了,稍微了解下吧):

  • 文件格式验证:验证字节流是否符合Class文件格式的规范。
  • 元数据验证:其实就是这个类对应.class文件的字节码描述的元数据信息是否符合Java规范。比如:是否存在父类,父类的继承链是否正确,抽象类是否实现了其父类或接口之中要求实现的所有方法,字段、方法是否与父类产生矛盾等。该校验更多是Java规范层面的校验。
  • 字节码验证:对类的方法体进行校验分析,确保这些方法在运行时是合法的、符合逻辑的。
  • 符号引用验证:发生在解析阶段。
    验证阶段不是必须执行的,该阶段可以跳过以缩短虚拟机类加载的时间。

准备

在该阶段会为类的静态字段信息(即被static修饰的变量)分配内存,并且设置初始值

  • 内存分配仅包括 static 修饰过的变量,而不包括实例变量,实例变量得等到对象实例化时分配内存。
  • 初始值指的是变量数据类型的默认值,而不是被在Java代码中被显式地赋予的值。但是,当字段信息被 final 修饰成常量时,这个初始值就是Java代码中显式地赋予的值。
  • 这些静态变量是存放在方法区内的,方法区是一个逻辑概念,其具体实现为jdk1.8的元空间与jdk1.8之前的永久代,JDK8以后取消了永久代改为元空间,而元空间在逻辑上存在物理上不存在,这些类变量的内存实际上是在堆内存中的。

解析

虚拟机会把这个Class文件中,常量池内的符号引用转换为直接引用。主要解析的是 类或接口、字段、类方法、接口方法、方法类型、方法句柄等符号引用。我们可以把解析阶段中,符号引用转换为直接引用的过程,理解为当前加载的这个类,和它所引用的类,正式进行“连接“的过程。

符号引用与直接引用

符号引用就是一个类中(当然不仅是类,还包括类的其他部分,比如方法,字段等),引入了其他的类,可是JVM并不知道引入的其他类在哪里,所以就用唯一符号来代替,等到类加载器去解析的时候,就把符号引用找到那个引用类的地址,这个地址也就是直接引用。

通俗的话来讲就是:

  • 符号引用其实可以理解为一个唯一标识,可以是任意值,只要能通过这个值能定位到目标。
  • 直接引用就是可以直接或间接指向目标内存位置的指针或句柄。

初始化

这个步骤是不是与bean生命周期的初始化阶段名称相同?但是其实是完全不同的处理阶段。类加载的初始化过程,就是执行类构造器 ()方法的过程。
类加载初始化完成后,类中static修饰的变量会赋予程序员实际定义的“值”,同时类中如果存在static代码块,也会执行这个静态代码块里面的代码。

init和clinit区别
  • 最根本的区别就是:init是实例构造器,对非静态变量解析初始化,而clinit是class类构造器对静态变量,静态代码块进行初始化。
  • () 方法由编译器自动生成,但不是必须生成的,只有这个类存在static修饰的变量,或者类中存在静态代码块但时候,才会自动生成()方法。它是由编译器按语句在源文件中出现的顺序,依次自动收集类中的所有类变量的赋值动作和静态代码块中的语句合并产生的(不包括构造函数中的语句)。
  • 如下代码:
class Phli {static Log log = LogFactory.getLog(); // <clinit>private int x = 1;   // <init>X(){// <init>}static {// <clinit>}
}

类加载器

先看一张图:
在这里插入图片描述
从图中可以看出:

  • .class文件经过类加载器加载,其实也就是通过类全限定名来获取描述此类的二进制字节流的过程,而这正是类加载器在加载阶段要完成的动作。
  • 在堆中存放生成的Class类模板,后续就可以通过模板new不同的实例。
  • 类实例可以通过getClass获得Class模板,不同实例getClass获得的模板都是相同的。
  • Class模板可以通过getClassLoader获取其对应的类加载器。
  • 类加载器实现的功能,要完成的动作就是我们加载要完成的动作,在次不做赘述。

贴个main方法代码,印证下以上逻辑:

public class Phli {public String name;public static void main(String[] args) {Phli phli1 = new Phli();Phli phli2 = new Phli();Phli phli3 = new Phli();System.out.println(phli1.hashCode());System.out.println(phli2.hashCode());System.out.println(phli3.hashCode());Class<? extends Phli> aClass1 = phli1.getClass();Class<? extends Phli> aClass2 = phli2.getClass();Class<? extends Phli> aClass3 = phli3.getClass();System.out.println(aClass1.hashCode());System.out.println(aClass2.hashCode());System.out.println(aClass3.hashCode());}
}

执行结果:
在这里插入图片描述
很明显不同实例的Class模板是同一个。

三种类加载器

包括三类加载器(也可以说是四类,多了一个自定义加载器):启动类加载器(根加载器)、扩展类加载器及应用程序类加载器。这三种类加载器是分层的,类似于上下级关系(parent-child关系)。

  • 启动类加载器(Bootstrap):负责加载<JAVA_HOME>\lib 目录,或者被 -Xbootclasspath 参数制定的路径,例如 jre/lib/rt.jar 里所有的class文件。由C++实现,不是ClassLoader子类。
  • 扩展类加载器(Ext):负责加载Java平台中扩展功能的一些jar包,包括<JAVA_HOME>\lib\ext 目录中 或 java.ext.dirs 指定目录下的jar包。由Java代码实现。
  • 应用程序类加载器(App):负责加载ClassPath路径下所有jar包。

验证下,在刚才代码基础上在添加以下代码:

ClassLoader classLoader1 = aClass1.getClassLoader();
System.out.println(classLoader1);
ClassLoader classLoader2 = classLoader1.getParent();
System.out.println(classLoader2);
ClassLoader classLoader3 = classLoader2.getParent();
System.out.println(classLoader3);

执行结果:

sun.misc.Launcher$AppClassLoader@b4aac2
sun.misc.Launcher$ExtClassLoader@1615099
null

可以看到classLoader1为AppClassLoader也就是应用类加载器,然后调用它的getParent方法,得到的是ExtClassLoader也就是扩展类加载器,然后我们继续往上获取,结果为null。为什么为null,不应该是启动类加载器吗?
其实很简单,因为启动类加载器是C++实现的,Java程序获取不到~

双亲委派

在此之前我们先看一段代码:

package java.lang;public class String {public void test() {}public static void main(String[] args) {String string = new String();System.out.println("main");}
}

我们定义了一个package为java.lang的String类,然后其中包括main方法,执行看下结果:
在这里插入图片描述
what??明明有main方法为什么错误信息提示找不到main方法呢?这是因为双亲委派导致的。
双亲委派其实很简单,下面一句话就可以解释清楚:

任何一个类加载器在接到一个类的加载请求时,都会先让其父类进行加载,只有父类无法加载(或者没有父类)的情况下,才尝试自己加载

为什么没找到main方法,因为应用程序类加载器向上让扩展类加载器进行加载,扩展类根据全限定名没加载到,继续往上让启动类加载器加载,启动类加载了我们rt.jar里面的String了,肯定是没有main方法的。
在这里插入图片描述
这就是双亲委派,总结一下:

  1. 类加载器收到类加载的请求;
  2. 将这个请求向上委托给父类加载器去加载,一直向上委托,直到启动类加载器(Bootstrap);
  3. 启动类加载器检查是否能够加载当前类,能加在就结束,使用当前加载器,否则抛出异常,通知子加载器进行加载。
  4. 重复步骤3。

好处

使用双亲委派最大的作用就是:安全,可以保证Java核心类的API不会被随意篡改。

Native

这个更多的是了解是干什么的,熟悉概念。

概念

native首先是个关键字,是一个计算机函数,一个Native Method就是一个Java调用非Java代码的接口。方法的实现由非Java语言实现,比如C或C++。好,概念问题了解到这儿就可以了。

凡是用到native关键字的,就说明Java的作用范围达不到了。
其实平时的业务开发基本用不到,但是它是无处不在的,举几个例子:

  • Object基类中的方法:
public final native Class<?> getClass();
public native int hashCode();
protected native Object clone() throws CloneNotSupportedException;
  • Thread.start()
    在这里插入图片描述

JNI

JNI是Java Native Interface的缩写,通过使用 Java本地接口书写程序,可以确保代码在不同的平台上方便移植。 [1]
从Java1.1开始,JNI标准成为java平台的一部分,它允许Java代码和其他语言写的代码进行交互。JNI一开始是为了本地已编译语言,尤其是C和C++而设计的,但是它并不妨碍你使用其他编程语言,只要调用约定受支持就可以了。使用java与本地已编译的代码交互,通常会丧失平台可移植性。但是,有些情况下这样做是可以接受的,甚至是必须的。例如,使用一些旧的库,与硬件、操作系统进行交互,或者为了提高程序的性能。JNI标准至少要保证本地代码能工作在任何Java
虚拟机环境。

通过 JNI,我们就可以通过 Java 程序(代码)调用到操作系统相关的技术实现的库函数,从而与其他技术和系统交互,使用其他技术实现的系统的功能;同时其他技术和系统也可以通过 JNI 提供的相应原生接口开调用 Java 应用系统内部实现的功能。

相关文章:

【快速入门】JVM之类加载机制与Native

感慨&#xff1a; 如何定义一个合格的Java程序员&#xff0c;Java程序员要了解掌握哪些知识点&#xff0c;网上的面试题太多了&#xff0c;后端需要了解掌握的知识点太多太多了&#xff0c;Java基础、数据结构、异常、多线程、Spring、Spring boot、事务、算法、数据库&#xf…...

R实现数据分布特征的视觉化——多笔数据之间的比较

大家好&#xff0c;我是带我去滑雪&#xff01; 如果要对两笔数据或者多笔数据的分布情况进行比较&#xff0c;Q-Q图、柱状图、星形图都是非常好的选择&#xff0c;下面开始实战。 &#xff08;1&#xff09;绘制Q-Q图 首先导入数据bankwage.csv文件&#xff0c;该数据集…...

TCPUDP

TCP 1.什么是TCP TCP是处于运输层的通信协议&#xff0c;该协议能够实现数据的可靠性传输。 2.TCP报文格式 源端口和目的端口&#xff1a;各占两个字节&#xff0c;发送进程的端口和接收进程的端口号。 序号&#xff1a;占4个字节,序号如果增加到溢出&#xff0c;则下一个序…...

设计模式 - 备忘录模式

目录 一. 前言 二. 实现 三. 优缺点 一. 前言 备忘录模式又称快照模式&#xff0c;是一种行为型设计模式。它可以在不破坏封装性的前提下捕获一个对象的内部状态&#xff0c;并在对象之外保存这个状态&#xff0c;以便在需要的时候恢复到原先保存的状态。在不违反封装的情况…...

OpenCV4(C++)—— 几何图形的绘制

文章目录 一、基本图形1、线2、线圆3、线椭圆4、矩形 二、多边形 一、基本图形 1、线 绘制线&#xff0c;要给出两个点坐标 void cv::line(InputOutputArray img, Point pt1, Point pt2, const Scalar& color, int thickness 1, int lineType LINE_8, int shift 0);…...

智能优化算法常用指标一键导出为EXCEL,CEC2017函数集最优值,平均值,标准差,最差值,中位数,秩和检验,箱线图...

声明&#xff1a;对于作者的原创代码&#xff0c;禁止转售倒卖&#xff0c;违者必究&#xff01; 之前出了一篇关于CEC2005函数集的智能算法指标一键统计&#xff0c;然而后台有很多小伙伴在询问其他函数集该怎么调用。今天采用CEC2017函数集为例&#xff0c;进行展示。 为了突…...

python文件打包方式汇总

在Python中&#xff0c;你可以使用多种方法来打包你的项目&#xff0c;以下是最常见的两种方式&#xff1a; 使用PyInstaller&#xff1a; PyInstaller是一个非常实用的工具&#xff0c;可以将Python程序打包成独立的可执行文件。这样&#xff0c;你就可以在没有Python环境的…...

基于ChatGPT+词向量/词嵌入实现相似商品推荐系统

最近一个项目有个业务场景是相似商品推荐&#xff0c;给一个商品描述(比如 WIENER A/B 7IN 5/LB FZN )&#xff0c;系统给出商品库中最相似的TOP 5种商品&#xff0c;这种单纯的推荐系统用词向量就可以实现&#xff0c;不过&#xff0c;这个项目特点是商品库巨大&#xff0c;有…...

虾皮商品链接获取虾皮商品详情数据(用 Python实现虾皮商品信息抓取)

在网页抓取方面&#xff0c;可以使用 Python、Java 等编程语言编写程序&#xff0c;通过模拟 HTTP 请求&#xff0c;获取虾皮网站上的商品页面。在数据提取方面&#xff0c;可以使用正则表达式、XPath 等方式从 HTML 代码中提取出有用的信息。值得注意的是&#xff0c;虾皮网站…...

【数据库系统概论】数据查询之单表查询。详细解释WHERE、OEDER BY、GROUP BY 和 HAVING

前言 ❓单表查询选择表中的若干列查询经过计算的值选择表中的若干元组&#xff08;行&#xff09;消除取值重复的行查询满足条件的元组&#xff08;WHERE&#xff09; 对查询结果排序&#xff08;ORDER BY&#xff09;聚集函数对查询结果分组&#xff08;GROUP BY&#xff09; …...

2023年医药商业行业发展研究报告

第一章 行业概况 1.1 定义 医药商业行业&#xff0c;作为医药领域的重要组成部分&#xff0c;扮演着至关重要的角色。这一行业专注于医药商品的经营与流通&#xff0c;确保药品能够有效、安全地到达消费者手中。随着医药科技的进步和市场需求的增长&#xff0c;医药商业行业在…...

Android 消息机制

Android 消息机制 Android 的消息机制也是Handler机制&#xff0c;主要作用是用来在不同线程之间通信&#xff0c;通常使用在子线程执行完成一些儿耗时操作&#xff0c;需要回到主线程更新UI时&#xff0c;通过Handler将有关UI操作切换到主线程。由于Android中主线程不可进行耗…...

QT计时器QTime的使用举例

Qt 中的计时器&#xff08;QTimer&#xff09;是一种用于执行定时操作的机制。您可以使用 QTimer 来执行周期性任务、在一段时间后执行操作或创建间隔定时器。以下是使用 QTimer 的基本步骤以及一个简单的示例&#xff1a; **包含头文件&#xff1a;**首先&#xff0c;确保您的…...

js中await用法

在JavaScript中&#xff0c;await用于暂停异步函数执行&#xff0c;等待Promise对象的解决。当Promise对象解决时&#xff0c;await将返回被解决的值&#xff0c;否则它将抛出一个被拒绝的Promise错误。 下面是一些使用await的例子&#xff1a; 使用await等待一个Promise对象…...

Qt多工程同名字段自动翻译工具

开发背景 项目里不同工程经常会引用同一批公共类&#xff0c;这些类里如果有字段需要翻译&#xff0c;需要在不同的项目里都翻译一遍&#xff0c;比较麻烦冗余。 特此开发了这个小翻译工具&#xff0c;能读取程序目录下的所有ts文件&#xff0c;以类名归类&#xff0c;不同项目…...

vue3+elementui实现表格样式可配置

后端接口传回的数据格式如下图 需要依靠后端传回的数据控制表格样式 实现代码 <!-- 可视化配置-表格 --> <template><div class"tabulation_main" ref"myDiv"><!-- 尝试过在mounted中使用this.$refs.myDiv.offsetHeight,获取父元素…...

x11截屏源码(ubuntu18.04)

使用x11库实现截屏并保存为png图片 【shot.c】 // filename: shot.c #include <X11/Xlib.h> #include <X11/Xutil.h> #include <X11/Xatom.h> #include <X11/cursorfont.h> #include <png.h> #include <stdio.h> #include <stdlib.h>…...

【ComfyUI】MacBook Pro 安装(Intel 集成显卡)

文章目录 环境概述配置pip镜像配置pip代理git配置&#xff08;选配&#xff09;下载comfyUI代码创建、激活虚拟环境下载依赖安装torchvision启动comfyUI为什么Mac不支持CUDA&#xff0c;即英伟达的显卡&#xff1f;安装Intel工具包 环境 显卡&#xff1a;Intel Iris Plus Grap…...

HTTPS 加密全过程

加密协议以前是SSL,现在都是TLS, 而证书现在大多数都是SSL证书 抓包流程: TCP三次握手过后, 客户端发送Client Hello 服务器相应Server Hello 服务器再次响应发送证书: 服务器再发送公钥:...

联邦学习综述二

联邦学习漫画 联邦学习漫画链接: https://federated.withgoogle.com/ Federated Analytics: Collaborative Data Science without Data Collection 博客链接: https://blog.research.google/2020/05/federated-analytics-collaborative-data.html 本篇博客介绍了联邦分析&a…...

Idea本地跑flink任务时,总是重复消费kafka的数据(kafka->mysql)

1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 Idea中执行任务时&#xff0c;没法看到JobManager的错误&#xff0c;以至于我以为是什么特殊的原因导致任务总是反复消费。在close方法中&#xff0c;增加日志&#xff0c;发现jdbc连接被关闭了。 重新…...

基于nodemailer实现邮件发送

概述 node中可用nodemailer实现邮件的发送。本文使用QQ邮箱实现邮件的发送。 实现效果 实现 1. QQ邮箱配置 首先需要开启POP3/IMAP/SMTP/Exchange/CardDAV/CalDAV服务&#xff0c;如下图所示。 生成授权码 2. 发送邮件 发送邮件的代码比较简单&#xff0c;如下&#xf…...

【PostgreSQL内核学习(十八)—— (数据库表参数)】

数据库表参数 default_reloptions 函数案例 声明&#xff1a;本文的部分内容参考了他人的文章。在编写过程中&#xff0c;我们尊重他人的知识产权和学术成果&#xff0c;力求遵循合理使用原则&#xff0c;并在适用的情况下注明引用来源。 本文主要参考了《PostgresSQL数据库内核…...

区块链的两个核心概念之一签名, 另一个是共识.

Alice的公私钥&#xff0c; 签名和验证签名仅仅确定了Alice对数字资产A所有权的宣言. 之后, Bob也可以用自己的私钥对资产A进行签名宣誓所有权。区块链中叫双花&#xff0c;即重复宣称所有权&#xff0c; 也称重复花费交易。这时候需要共识算法(集体成员pow或委员会代表pos监督…...

wpf中prism框架切换页面

主页面...

正则表达式(Regular Expression)学习网址分享

正则表达式&#xff08;Regular expressions&#xff0c;也叫REs、 regexs 或regex patterns&#xff09;&#xff0c;是一种文本模式&#xff0c;包括普通字符&#xff08;例如&#xff0c;a 到z 之间的字母&#xff09;和特殊字符&#xff08;称为"元字符"&#xf…...

【已解决】socket.gaierror: [Errno -3] Temporary failure in name resolution

问题描述 今天在环境迁移的过程中遇到多个问题&#xff0c;包括ModuleNotFoundError: No module named flask&#xff0c;socket.gaierror: [Errno -3] Temporary failure in name resolution以及Downloading: "https://huggingface.co/gyrojeff/YuzuMarker.FontDetection…...

CUDA code=700(cudaErrorIllegalAddress) 报错与排查方法

CUDA code700(cudaErrorIllegalAddress) 报错与排查方法 最近笔者在调试自己写的 CUDA 代码时, 遇到了 code700(cudaErrorIllegalAddress) 的报错, 在此记录一下排查和解决方法. 报错 报错是由 CUDA API 函数执行时产生的, 由 checkCudaErrors() 函数检测出(CUDA 常用错误检…...

项目管理过程组

项目管理有2条主线&#xff0c;一条是技术&#xff0c;一条是管理。项目过程由项目团队实施。一般术语以下两大类之一&#xff1a;一类是项目管理过程。另一类是面向产品的过程。在大多数情况下&#xff0c;大多数项目都有共同的项目管理过程。它们通过有目的的实施而互相联系起…...

python每日一练(5)

&#x1f308;write in front&#x1f308; &#x1f9f8;大家好&#xff0c;我是Aileen&#x1f9f8;.希望你看完之后&#xff0c;能对你有所帮助&#xff0c;不足请指正&#xff01;共同学习交流. &#x1f194;本文由Aileen_0v0&#x1f9f8; 原创 CSDN首发&#x1f412; 如…...