JVM:从零到入门

JVM,就是Java虚拟机。

JVM是一个巨大的话题,我们本文主要简单介绍一些围绕JVM相关的基础知识。

目录

JVM内存区域划分

本地方法栈

虚拟机栈

程序计数器

方法区/ 元数据区

类加载

1.加载

2.验证

3.准备

4.解析

5.初始化

双亲委派模型

垃圾回收机制

引用计数

可达性分析

如何清除垃圾

标记清除

复制算法

标记整理


JVM内存区域划分

JVM启动的时候会申请到一个很大的内存区域,根据需要,把整个空间分成好几个部分。

本地方法栈

native就表示JVM内部的C++代码,这个部分就是给调用native方法(JVM内部的方法)准备的空间。

虚拟机栈

JVM Stacks 是JVM中一个特定的空间,这里储存的是方法之间的调用关系。(前面的本地方法栈储存的就是native之间的调用关系)

Heap,是整个JVM中最大的区域。此内存区域的唯一目的就是存放对象实例,如类的成员变量等等。在Java虚拟机规范中的描述是:所有的对象实例以及数组都要在堆上分配。

程序计数器

对于线程来说,在cpu上运行的时候如果cpu被调度走了,那么怎么保证cpu回来的时候线程还是当前进度呢?程序计数器就是干的这个活,记录当前线程执行到哪个指令。为了线程切换后能恢复到正确的执行位置,每条线程都有一个独立的程序计数器,各个线程之间计数器互不影响,独立存储。

方法区/ 元数据区

Java8开始改名,以前叫做方法区,现在叫元数据区。

它用于存储已被Java虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等数据。

同时要注意,只有元数据区和堆上的数据是多线程共享的,其他的都是每个线程独立的数据。

局部变量在栈上,普通成员变量在堆上,静态成员变量在元数据区。

类加载

程序员编写一个.java文件后,javac把这个.java文件编译成.class文件,然后再通过JVM中的类加载,从硬盘被加载到内存中的过程。

1.加载

在加载 Loading 阶段,Java虚拟机需要完成以下三件事情:

1)通过一个类的全限定名来获取定义此类的二进制字节流。

2)将这个字节流所代表的静态存储结构转化为方法区的运行时数据结构。

3)在内存中生成一个代表这个类的java.lang.Class对象,作为方法区这个类的各种数据的访问入口。

也就是把文件读到内存中的步骤。

2.验证

验证是连接阶段的第一步,这一阶段的目的是确保Class文件的字节 流中包含的信息符合《Java虚拟机 规范》的全部约束要求,保证这些信 息被当作代码运行后不会危害虚拟机自身的安全。

比如:文件格式验证 字节码验证 符号引用验证

3.准备

准备阶段是正式为类中定义的变量(即静态变量,被static修饰的变量)分配内存并设置类变量初始值的阶段。

例如:

private static int value = 123;

此时就是初始化value的int值为0,而不是123。(相当于在元数据区先占个位置)

4.解析

解析阶段是 Java 虚拟机将常量池内的符号引用替换为直接引用的过程,也就是初始化常量的过程。

5.初始化

初始化阶段,Java 虚拟机真正开始执行类中编写的 Java 程序代码,将主导权移交给应用程序。初始化 阶段就是执行类构造器方法的过程。

那么一个类什么时候会被加载呢?

不是Java程序一运行就会把所有的类都加载,而是真正用到了才会加载。(懒汉模式)

1.构造类的实例

2.调用类的静态方法、使用静态属性

3.加载子类,就会先加载其父类

双亲委派模型

加载是把.class文件找到,读取文件内容的过程,双亲委派模型就是描述这个加载,找.class文件的基本过程。

JVM默认提供了三个类加载器:

上述三个类,存在parent属性,指向自己的父类加载器。

加载一个类的时候,先从ApplicationClassLoader开始的,但是它会把加载任务交给其父亲ExtensionClassLoader,让其父亲去执行。于是ExtensionClassLoader要去加载了,但是也不是真的加载,而是把加载任务再委托给自己的父亲。它的父亲BootstrapClassLoader要去加载了,但是发现自己的父亲是null,没有父亲,于是只能自己加载。

等待自己加载完后,如果找到了就加载,没有找到就由自己的子类加载器进行加载。也就是BootstrapClassLoader加载完没找到后,让ExtensionClassLoader加载,如果也没找到最后再让ApplicationClassLoader加载。

最后ApplicationClassLoader如果找到了就加载,没找到就由子类加载器进行加载。由于当前没有子类了,就只能抛出类找不到这样的异常。

为什么要有上述顺序?这套顺序出自JVM实现代码的逻辑,并且这段代码大概是类似于递归的方式写的。这个顺序最主要的目的就是为了保证Bootstrap能够先加载,Application后加载,这样能够解决用户自己创建了一些奇怪的类引起bug。

例如用户自己写了一个java.lang.String这样的类,按照上述加载流程,JVM此时加载道德其实是标准库的类,而不会加载到用户自己的这个类。

同时用户自己也可以定义类加载器,可以和现有的配合使用。

垃圾回收机制

我们在java开发中何时考虑过内存管理?不像c和c++还要考虑什么时候释放资源,java只需要考虑业务实现就行了。

什么是垃圾呢?垃圾指的是不再使用的内存,垃圾回收就是把不用的内存帮我们自动释放了。

在别的语言中,如果不手动释放垃圾,这块内存空间就会一直存在,一直存在到进程结束。这就可能导致内存泄漏。剩余的空间越来越少,进一步导致后续的内存申请操作失败!

GC是垃圾回收中最主流的一种方式。很多语言都使用GC来解决上述问题。

JVM中有很多内存区域,GC主要是针对堆进行释放的。

同时GC是以对象为基本单位进行回收的,而不是字节。要回收就是回收整个对象,而不是回收半个对象。

GC回收的是对象,那么就要看有没有引用指向它。

如何知道有没有引用指向呢?有两种方式:

1.引用计数

2.可达性分析

引用计数

引用计数中,给每个对象都分配了一个计数器(整数),每次创建一个引用指向这个对象计数器就+1,这个引用被销毁了计数器就-1。

这个办法简单有效,但是Java并没有使用。因为:

1.内存空间浪费很严重,一个对象就要分配一个计数器。

2.存在循环引用的问题。

所以在Java中不用引用计数这种方式。

可达性分析

Java中的对象都是通过引用来指向并访问的。所以整个Java中的所有对象类似于一个树根节点,通过链式或者树形结构一个个串连起来。

可达性分析就是把这些对象被组织的结构看做树,从树根节点出发,遍历树,所有能被访问到的对象标记成“可达”。

通过这种方式,只需要有一个GC Roots节点就可以访问到所有的元素。访问不到的就代表着可以被GC回收了。所有的对象在创建的时候JVM都会记录下地址。

1、虚拟机栈(栈帧中的本地变量表)中引用的对象
2、方法区中类静态属于引用的对象
3、方法区中常量引用的对象
4、本地方法栈中Native方法引用的对象。

如何清除垃圾

主要有三种方式:标记清除,复制算法,标记整理。

标记清除

在一块内存空间中,2,4,6是需要清楚的垃圾,那么直接对其垃圾回收就可以完成。但是这样做就会有一个问题:全都是内存碎片。被释放的空闲空间是零散的,不是连续的。如果要申请一块很大的空间,必须要是连续的空间,此时就会申请失败!

复制算法

复制算法会把内存分成两半,对于垃圾(1和3)不去处理,把不是垃圾的部分(2和4)复制到另一边。

此时再把左边的内存清除。这样也就是把需要回收的垃圾处理掉了。

这种方法空间利用率低,并且垃圾少、有效对象多那么复制成本就比较大。

标记整理

标记整理解决了复制算法的缺点,类似于顺序表删除中间元素的操作,并且会有元素搬运的操作。

1 3 5 是垃圾,在标记整理的时候,会把1 3 5删除,然后把2移动到1的位置,4移动到2的位置,6移动到3的位置。

这种方式保证了空间利用率,并且解决了内存碎片问题。

但是这种方式效率也不高,如果搬运的空间比较大开销也会很大。

实际上基于这一些基本的策略,把垃圾回收分成不同的场景,不同的场景有不同的算法。这就是“分代回收”。

这个“代”,就是一代代的代。我们给对象引入一个概念:年龄。这个年龄,是熬过GC的轮次。如果经历了一轮可达性分析,发现这个对象还不是垃圾,就说明这个对象熬过了一轮GC。

JVM把堆分成两个区域:新生代、老年代。

新生代又分为Eden、From Survivor、To Survivor。

  1. 生代 ( Young ) 与老年代 ( Old ) 的比例的值为 1:2 ( 该值可以通过参数 –XX:NewRatio 来指定 ),即:新生代 ( Young ) = 1/3 的堆空间大小。老年代 ( Old ) = 2/3 的堆空间大小。

  2. Edem : From Survivor : To Survivor = 8 : 1 : 1 ( 可以通过参数 –XX:SurvivorRatio 来设定 ),即: Eden = 8/10 的新生代空间大小,From Survivor = To Survivor = 1/10 的新生代空间大小。

数据会首先分配到Eden区当中(当然也有特殊情况,如果是大对象那么会直接放入到老年代(大对象是指需要大量连续内存空间的java对象)。当Eden没有足够空间的时候就会触发jvm发起一次Minor GC,。如果对象经过一次Minor-GC还存活,并且又能被Survivor空间接受,那么将被移动到Survivor空间当中。并将其年龄设为1,对象在Survivor每熬过一次Minor GC,年龄就加1,当年龄达到一定的程度(默认为15)时,就会被晋升到老年代中了,当然晋升老年代的年龄是可以设置的。

这也就是之前的三种垃圾回收机制的分代回收中的代。其实主要原因就是可以根据各个年代的特点进行对象分区存储,更便于回收,采用最适当的收集算法:

新生代中,每次垃圾收集时都发现大批对象死去,只有少量对象存活,便采用了复制算法,只需要付出少量存活对象的复制成本就可以完成收集。(从From Survivor到To Survivor用的是复制算法)

而老年代中因为对象存活率高、没有额外空间对它进行分配担保,就必须采用标记清除或者标记整理算法。

到此为止,JavaEE初阶的内容告一段落~接下来博主会持续更新JavaEE进阶一些的内容,快快关注吧~


 

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

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

相关文章

yydict属性字典-一种更加方便的方式访问字典

yydict属性字典-一种更加方便的方式访问字典 问题引入 这篇文章是想介绍 最近在使用字典的一种困惑. 我希望通过少写几个字符来访问 python中字典这种数据结构. 比如这个例子: person {name: frank,age: 18,hobby: swimming }在python中字典的定义 如上面的例子, 如果我希…

spring boot mybatis plus mapper如何自动注册到spring bean容器

##Import(AutoConfiguredMapperScannerRegistrar.class) ##注册MapperScannerConfigurer ##MapperScannerConfigurer.postProcessBeanDefinitionRegistry方法扫描注册mapper ##找到mapper候选者 ##过滤mapper 类 候选者 ##BeanDefinitionHolder注册到spring 容器

vue项目之.env文件.env.dev、test、pro

.env文件是vue运行项目时的环境配置文件。 .env: 全局默认配置文件,所有环境(开发、测试、生产等)均会加载并合并该文件 .env.development(开发环境默认命名) 开发环境的配置,文件名默认为.env.development,如果需要改名也是可以的&#xf…

【Python】使用pyinstaller打包为Windows平台的xxx.exe方法步骤

pyinstaller 是一个用于将 Python 代码打包成独立可执行文件的工具,它可以将 Python 代码打包成 Windows、Linux、Mac 等平台的可执行文件,方便用户在不同环境中运行。 pyinstaller用法: 1.安装pyinstaller库,这里以PyCharm环境为…

ACM论文LaTeX模板解析(一)| 模板下载与安装

本文收录于专栏:ACM 论文 LaTeX模板解析,本专栏将会围绕ACM 论文 LaTeX模板解析持续更新。欢迎点赞收藏关注! 文章目录 1. 引言2. 下载方式 1. 引言 计算机械协会(ACM,Association for Computing Machinery&#xff0…

UG装配-WAVE几何链接器

自上向下(自顶向下)设计 先将产品主要结构(或主要部件)建立好,然后再根据要求设计其它组件,使每个组件之间有数据关联,适用于产品开发初期,便于修改,修改组件数据后&…

基于JavaWeb的酒店管理系统

基于JavaWeb的酒店管理系统 文章目录 基于JavaWeb的酒店管理系统系统介绍技术选型成果展示源码获取账号地址及其他说明 系统介绍 基于JavaWeb的酒店管理系统是为酒店打造的管理平台,其主要功能有管理员登陆、客房预订、客房入住、房间管理、数据查询(预订单查询、入…

PySide6/PyQt6中的时间管理类:QTime的使用方法

文章目录 📖 介绍 📖🏡 环境 🏡📒 使用方法 📒📝 创建QTime对象📝 常用方法⚓️ 相关链接 ⚓️📖 介绍 📖 QTime是PySide6中用于处理时间段的类,可以用来表示一天中的时间,例如小时、分钟和秒。它提供了许多操作和格式化时间的功能,使得处理时间变得更加…

mysql-实战案例 (超详细版)

🎉欢迎您来到我的MySQL基础复习专栏 ☆* o(≧▽≦)o *☆哈喽~我是小小恶斯法克🍹 ✨博客主页:小小恶斯法克的博客 🎈该系列文章专栏:重拾MySQL 🍹文章作者技术和水平很有限,如果文中出现错误&am…

VL53L4CD TOF开发(1)----驱动TOF进行测距

VL53L4CX TOF开发.1--驱动TOF进行测距 概述视频教学样品申请完整代码下载主要特点硬件准备技术规格系统框图应用示意图生成STM32CUBEMX选择MCU串口配置IIC配置 XSHUTX-CUBE-TOF1演示结果 概述 VL53L4CD适用于接近测量和短距离测量,可实现从仅仅1 mm到1300 mm的超精…

SAP SD-DN-MM 交货单相关物料凭证的视图的日期问题

眼下有个需求 获取交货单对应的物料凭证的过账日期BLDAT。 同步BW数据过去 新增一个数据库视图 但是实际使用时,有效部分仅本月,再选择条件里面要加上 MATdoc-bldat > sy-datum - sydatum6(2). 于是使用ST05 跟踪了一下,发现在DD28S…

echarts——折线图实现不同区间不同颜色+下钻/回钻功能——技能提升

echarts——折线图实现不同区间不同颜色下钻/回钻功能——技能提升 需求场景解决步骤1:安装echarts插件解决步骤2:html代码解决步骤3:封装option配置和initChart渲染方法解决步骤4:回钻功能 需求场景 最近在写后台管理系统时&…

乐意购项目前端开发 #2

一、Axios的安装和简单封装 安装Axios npm install axios在utils目录下创建 http.js 文件, 内容如下 import axios from axios// 创建axios实例 const http axios.create({baseURL: http://localhost:9999,//后端服务器地址timeout: 5000 })// axios请求拦截器 http.interc…

为什么使用双token实现无感刷新用户认证?

单token机制 认证机制:对与单token的认证机制在我们项目中仅使用一个Access Token的访问令牌进行用户身份认证和授权的方案处理。 不足之处: 安全性较低(因为只有一个token在客户端和服务器端之间进行传递,一旦Access Token被截…

pip查看某个包存在的历史版本

简介:当我们想查看某个包有哪些可安装版本,但是又不想去官网查询,如何用pip命令查询出全部历史版本? 历史版本: Python:pip升级超时解决方案 Python:指定的Python版本pip Python&#xff1a…

鸿蒙开发-UI-布局

鸿蒙开发-序言 鸿蒙开发-工具 鸿蒙开发-初体验 鸿蒙开发-运行机制 鸿蒙开发-运行机制-Stage模型 鸿蒙开发-UI 鸿蒙开发-UI-组件 鸿蒙开发-UI-组件-状态管理 鸿蒙开发-UI-应用-状态管理 鸿蒙开发-UI-渲染控制 文章目录 前言 一、布局概述 1.布局结构 2.布局元素组成 3.布局分类 …

TRB 2024论文分享:基于生成对抗网络和Transformer模型的交通事件检测混合模型

TRB(Transportation Research Board,美国交通研究委员会,简称TRB)会议是交通研究领域知名度最高学术会议之一,近年来的参会人数已经超过了2万名,是参与人数和国家最多的学术盛会。TRB会议几乎涵盖了交通领域…

三、ngxin虚拟主机

目录 什么是nginx虚拟主机修改端口 访问页面1、配置nginx.config 文件2、 添加配置给目录中写入内容检测nginx 是否有语法错误(nginx -t)重启 nginx查看配置结果 不同主机网卡 查看到不同的页面先添加一个临时ip修改ngixn配置文件创建目录文件检测nginx …

React入门 - 06(TodoList 列表数据的新增和删除)

本章内容 目录 一、实践一下 React 的列表渲染二、TodoList 新增功能三、列表循环的 key四、删除 上一节内容我们完成了输入框中可以自由输入内容,这一节我们继续 TodoList功能的完善:列表数据的新增和删除。 在开始之前,我们先介绍一下 Re…

前端对接电子秤、扫码枪设备serialPort 串口使用教程

因为最近工作项目中用到了电子秤,需要对接电子秤设备。以前也没有对接过这种设备,当时也是一脸懵逼,脑袋空空。后来就去网上搜了一下前端怎么对接,然后就发现了SerialPort串口。 Serialport 官网地址:https://serialpo…