JVM(内存划分,类加载,垃圾回收)
JVM
Java程序,是一个名字为Java 的进程,这个进程就是所说的“JVM”
1.内存区域划分
JVM会先从操作系统这里申请一块内存空间,在这个基础上再把这个内存空间划分为几个小的区域
在一个JVM进程中,堆和方法区只有一份;栈和程序计数器,每个线程有一份

1.堆:存放new的对象
堆里面分为两个区域:新生代和老年代,新生代放新建的对象,当经过一定GC次数之后还存活的对象会放入老年代。新生代还有三个区域:一个 Endn + 两个 Survivor(S0/S1)

垃圾回收的时候会将 Endn 中存货的对象放到一个未使用的Survivor 中,并把当前的 Endn 和正在使用的 Survivior 清除掉
2.栈:存放方法之间的调度关系
虚拟机栈:java里面用来保存调用关系的内存空间
本地方法栈:本地方法,就是JVM内部 C++ 写的代码,调用关系的内存空间
3.程序计数器:存放下一个要执行的指令的地址
程序计数器是一块比较小的内存空间,可以看作释放前线程所执行的字节码的行号指示器
4.方法区:存放类对象(加载好的)
方法区的作用:用来存储被虚拟机加载的类信息,常量,静态变量,即时编译器编译后的代码等数据的
变量在哪个部分和变量类型无关,和变量的形态有关(成员,静态,局部)
代码里的局部变量:栈
代码里的成员变量:堆
代码里的静态变量:方法区
2.类加载
java程序在运行之前,需要先编译 .java => .class(二进制字节码文件)运行的时候,Java 进程(JVM)就会读取对应的 .class 文件,并且解析内容,在内存中构造出类对象并进行初始化
1.类加载过程

其中前5步时固定的顺序,并且也是类加载的过程,其中中间的3步都属于连接,所以类加载总共分为现在几个步骤
加载:找到.class文件,读取文件内容,并且按照.class规范格式来解析
验证:检查看当前的.class里的内容格式是否符合要求
准备:给类里的静态变量分配内存空间
解析:初始化字符串常量,把符号引用替换成直接引用(在类加载之前,字符串常量是没有分配内存空间的,变量名称无法保存字符串常量的真实地址,只能先使用一个占位符,等类加载完成,字符串常量分配过内存之后,就可以用真正的地址替换占位符)
初始化:针对类进行初始化,初始化静态成员,执行静态代码块,并且加载父类
2.何时触发类加载
使用到一个类的时候,就触发加载(类并不一定是程序一启动就加载了,而是第一次使用才加载)
创建这个类的实例
使用了这个类的静态方法/静态属性
使用类的子类(加载子类就会触发加载父类)
3.双亲委派模型
如果一个类加载器收到了类加载的请求,它首先不会自己尝试去加载,而是将这个请求委派给父类加载器去完成,每一个层次的类加载器都是如此,因此所有的加载请求最终都应该传送到最顶层的启动类加载器中,只有当父加载器反馈自己无法完成这个加载请求时,子加载器才会尝试自己去完成加载
类加载器:JVM加载类,是由类加载器(class loader)模块来负责的
JVM 自带了多个类加载器:
Bootstrap ClassLoader 负责加载标准库中的类
Extension ClassLoader 负责加载 JVM 扩展的库的类
Application ClassLoader 负责加载自己的项目里的自定义类
双亲委派模型的工作过程:

上述三个类加载器存在父子关系
进行类加载的时候,输入的内容全限定类名,形如:java.lang.Thread
加载的时候,从Application ClassLoader开始
某个类加载器开始加载到时候,不会立即扫描自己的路径,而是先把任务委派给父“类加载器”
找到最上面的 Bootstrap ClassLoader在往上,没有父“类加载器”了,就会自己加载
如果父”类加载器“没有找到类,就会交给自己的子”类加载器“,继续加载
如果一直找到最下面的Application ClassLoader 也没有找到类,就会抛出一个“类没找到”异常,类加载就失败了
3.垃圾回收(GC)
Java堆中存放着几乎所有的对象实例,垃圾回收器在对堆进行垃圾回收前,首先要判断这些对象哪些还被引用着,哪些已经没有引用了。判断对象是否被引用有如下几种算法:
内存VS对象
在Java中,所有的对象都是要存在内存中的,因此将内存回收也叫做死亡对象的回收。
1.GC回收的部分
JVM主要内存分成这几个部分:
堆:GC主要针对堆来回收
方法区:类对象,加载之后也不太会卸载
栈:释放时机确定,不必回收
程序计数器:固定内存空间,不必回收

2.回收对象的判断算法
1.引用计数算法
给每个对象都加上一个计数器,这个计数器就表示“当前的对象几个引用”
但是,主流的 JVM 中没有选用引用计数法来管理内存,最主要的原因就是引用计数法无法解决对象循环引用问题
循环引用:
当两个对象互相引用时
classTest{
Testx;
}
Testa=newTest();
Testb=newTest();
a.x=b;
b.x=a;
a=null;
b=null;
当前这两个对象的计数器都为1,所以无法进行释放,此时外界的代码仍然时无法访问和使用对象的
2.可达性分析【JVM采取的方法】
核心思想:通过一系列称为 “GC Roots” 的对象作为起始点,从这些节点开始向下搜索,搜索走过的路径称之为 “引用链”,当一个对象到GC Roots没有任何的引用链时,证明此对象是不可用的

对象 Object 5-Object 7 之间虽然还有关联,但是它们到达不了GC Roots 是不可达的,因此他们会被判定为可回收对象
在Java语言中,课作为GC Roots的对象包含下面几种:
虚拟机栈(栈帧中的本地变量表)中引用的对象
方法区中类静态属性引用的对象
方法区中常量引用的对象
本地方法栈中JNI(Native)引用的对象
3.垃圾回收算法
1.标记-清除算法
首先标记出垃圾,然后直接把对象对应的内存空间进行释放

标记-清楚算法的不足:
效率问题:标记和清楚这两个过程的效率都不高
空间问题:标记清楚后会产生大量不连续的内存碎片,内存碎片较多会导致在分配内存空间较大的对象时,无法找到足够连续的内存
2.复制算法
复制算法是为了解决“标记-清楚”算法带来的问题描述:将一个内存空间分为两部分,首先先使用一边,清理时不再是原地释放,而是把“非垃圾”拷贝了另一边,然后再把之前这一边给释放掉

复制算法的不足:
空间利用率更低了(一次只能使用一半的内存空间)
如果一轮GC下来,大部分对象仍然需要保留,只有少数对象要回收,这个时候拷贝开销就很大
3.标记-整理算法
前面与“标记-清理”算法过程一致,打包后续不是直接堆可回收对象进行清理,而是让所有存活对象都想一段移动,然后直接清理掉边界以外的内存(类似于顺序表的删除元素操作)

这种方式,相对于上述的复制算法来说,空间利用率上来了,同时能够解决内存碎片问题,但是搬运操作也是比较耗时的
4.分代回收算法
分代算法是通过区域划分,实现不同区域和不同的垃圾回收策略,从而实现更好的垃圾回收(当前 JVM 垃圾收集采用此算法)一般是将Java堆分为新生代(GC 扫描的频率更高)和老年代(GC 扫描的频率降低),在新生代中,每次垃圾回收都有大批对象死去,只有少量存活,因此我们采用复制算法;而老年代中对象存活率高,没有额外空间对它进行分配担保,就必须采用“标记-清除”或者“标记-整理”算法
根据对象的不同特点(根据对象的年龄(依据 GC 的轮次来算的,有一组线程,周期性的扫描代码里所有的对象,如果一个对象,经历了一次GC ,就认为年龄+1)来划分)。
一个基本的经验规律:如果一个对象的寿命比较长,大概率还会活很久

上述规则还有个特殊情况,如果对象是一个非常大的对象,将会直接进入老年代!
相关文章:
JVM(内存划分,类加载,垃圾回收)
JVMJava程序,是一个名字为Java 的进程,这个进程就是所说的“JVM”1.内存区域划分JVM会先从操作系统这里申请一块内存空间,在这个基础上再把这个内存空间划分为几个小的区域在一个JVM进程中,堆和方法区只有一份;栈和程序…...
工作中遇到的问题 -- 你见过哪些写的特别好的代码
strPtr : uintptr((*(*stringStruct)(unsafe.Pointer(&str))).str)代码解析: 这是一段 Go 代码,它的作用是获取一个字符串变量 str 的底层指针,即字符串数据的起始地址。 这段代码涉及到了 Go 语言中的指针、类型转换和内存布局等概念&…...
基于chatGPT设计卷积神经网络
1. 简介 本文主要介绍基于chatGPT,设计一个针对骁龙855芯片设计的友好型神经网络。 提问->跑通总共花了5min左右,最终得到的网络在Cifar100数据集上与ResNet18的精度对比如下。 模型flopsparamstrain acc1/5test acc1/5ResNet18(timm)1.8211.18~98…...
java.sql.Date和java.util.Date的区别
参考答案 java.sql.Date 是 java.util.Date 的子类java.util.Date 是 JDK 中的日期类,精确到时、分、秒、毫秒java.sql.Date 与数据库 Date 相对应的一个类型,只有日期部分,时分秒都会设置为 0,如:2019-10-23 00:00:0…...
动态规划---线性dp和区间dp
动态规划(三) 目录动态规划(三)一:线性DP1.数字三角形1.1数字三角形题目1.2代码思路1.3代码实现(正序and倒序)2.最长上升子序列2.1最长上升子序列题目2.2代码思路2.3代码实现3.最长公共子序列3.1最长公共子序列题目3.2代码思路3.3代码实现4.石子合并4.1题目如下4.2代…...
常见的2D与3D碰撞检测算法
分离轴分离轴定理(Separating Axis Theorem)是用于解决2D或3D物体碰撞检测问题的一种方法。其基本思想是,如果两个物体未发生碰撞,那么可以找到一条分离轴(即一条直线或平面),两个物体在该轴上的…...
STM32 10个工程篇:1.IAP远程升级(二)
一直提醒自己要更新CSDN博客,但是确实这段时间到了一个项目的关键节点,杂七杂八的事情突然就一涌而至。STM32、FPGA下位机代码和对应Labview的IAP升级助手、波形设置助手上位机代码笔者已经调试通过,因为不想去水博客、凑数量,复制…...
Unity+ChatGpt的联动 AICommand
果然爱是会消失的,对吗 chatGpt没出现之前起码还看人家的文章,现在都是随便你。 本着师夷长技以制夷的思路,既然打不过,那么我就加入 github地址:https://github.com/keijiro/AICommand 文档用chatGpt翻译如下&#…...
STM-32:按键控制LED灯 程序详解
目录一、基本原理二、接线图三、程序思路3.1库函数3.2程序代码注:一、基本原理 左边是STM322里电路每一个端口均可以配置的电路部分,右边部分是外接设备 电路图。 配置为 上拉输入模式的意思就是,VDD开关闭合,VSS开关断开。 浮空…...
北邮22信通:(8)实验1 题目五:大整数加减法(搬运官方代码)
北邮22信通一枚~ 跟随课程进度每周更新数据结构与算法的代码和文章 持续关注作者 解锁更多邮苑信通专属代码~ 上一篇文章: 北邮22信通:(7)实验1 题目四:一元多项式(节省内存版)_青山如…...
Fiddler抓取https史上最强教程
有任何疑问建议观看下面视频 2023最新Fiddler抓包工具实战,2小时精通十年技术!!!对于想抓取HTTPS的测试初学者来说,常用的工具就是fiddler。 但是初学时,大家对于fiddler如何抓取HTTPS难免走歪路ÿ…...
STM32开发基础知识入门
C语言基础 位操作 对基本类型变量可以在位级别进行操作。 1) 不改变其他位的值的状况下,对某几个位进行设值。 先对需要设置的位用&操作符进行清零操作,然后用|操作符设值。 2) 移位操作提高代码的可读性。 3) ~取反操作使用技巧 可用于对某…...
学习操作系统的必备教科书《操作系统:原理与实现》| 文末赠书4本
使用了6年的实时操作系统,是时候梳理一下它的知识点了 摘要: 本文简单介绍了博主学习操作系统的心路历程,同时还给大家总结了一下当下流行的几种实时操作系统,以及在工程中OSAL应该如何设计。希望对大家有所启发和帮助。 文章目录…...
大数据的常用算法(分类、回归分析、聚类、关联规则、神经网络方法、web数据挖掘)
在大数据时代,数据挖掘是最关键的工作。大数据的挖掘是从海量、不完全的、有噪声的、模糊的、随机的大型数据库中发现隐含在其中有价值的、潜在有用的信息和知识的过程,也是一种决策支持过程。其主要基于人工智能,机器学习,模式学…...
【数据结构】详解二叉树与堆与堆排序的关系
🌇个人主页:平凡的小苏 📚学习格言:别人可以拷贝我的模式,但不能拷贝我不断往前的激情 🛸C语言专栏:https://blog.csdn.net/vhhhbb/category_12174730.html 🚀数据结构专栏ÿ…...
【Pandas】数据分析入门
文章目录前言一、Pandas简介1.1 什么是Pandas1.2 Pandas应用二、Series结构2.1 Series简介2.2 基本使用三、DataFrame结构3.1 DataFrame简介3.2 基本使用四、Pandas-CSV4.1 CSV简介4.2 读取CSV文件4.3 数据处理五、数据清洗5.1 数据清洗的方法5.2 清洗案例总结前言 大家好&…...
【c++】:list模拟实现“任意位置插入删除我最强ƪ(˘⌣˘)ʃ“
文章目录 前言一.list的基本功能的使用二.list的模拟实现总结前言 1. list是可以在常数范围内在任意位置进行插入和删除的序列式容器,并且该容器可以前后双向迭代。2. list的底层是双向链表结构,双向链表中每个元素存储在互不相关的独立节点中࿰…...
QT表格控件实例(Table Widget 、Table View)
欢迎小伙伴的点评✨✨,相互学习🚀🚀🚀 博主🧑🧑 本着开源的精神交流Qt开发的经验、将持续更新续章,为社区贡献博主自身的开源精神👩🚀 文章目录前言一、图示实例二、列…...
第二章Vue组件化编程
文章目录模块与组件、模块化与组件化模块组件模块化组件化Vue中的组件含义非单文件组件基本使用组件注意事项使用 kebab-case使用 PascalCase组件的嵌套模板templateVueComponent一个重要的内置功能单文件组件Vue脚手架使用Vue CLI脚手架先配置环境初始化脚手架分析脚手架结构实…...
面试官:vue2和vue3的区别有哪些
目录 多根节点,fragment(碎片) Composition API reactive 函数是用来创建响应式对象 Ref toRef toRefs 去除了管道 v-model的prop 和 event 默认名称会更改 vue2写法 Vue 3写法 vue3组件需要使用v-model时的写法 其他语法 1. 创…...
2025年能源电力系统与流体力学国际会议 (EPSFD 2025)
2025年能源电力系统与流体力学国际会议(EPSFD 2025)将于本年度在美丽的杭州盛大召开。作为全球能源、电力系统以及流体力学领域的顶级盛会,EPSFD 2025旨在为来自世界各地的科学家、工程师和研究人员提供一个展示最新研究成果、分享实践经验及…...
解决Ubuntu22.04 VMware失败的问题 ubuntu入门之二十八
现象1 打开VMware失败 Ubuntu升级之后打开VMware上报需要安装vmmon和vmnet,点击确认后如下提示 最终上报fail 解决方法 内核升级导致,需要在新内核下重新下载编译安装 查看版本 $ vmware -v VMware Workstation 17.5.1 build-23298084$ lsb_release…...
高防服务器能够抵御哪些网络攻击呢?
高防服务器作为一种有着高度防御能力的服务器,可以帮助网站应对分布式拒绝服务攻击,有效识别和清理一些恶意的网络流量,为用户提供安全且稳定的网络环境,那么,高防服务器一般都可以抵御哪些网络攻击呢?下面…...
基于matlab策略迭代和值迭代法的动态规划
经典的基于策略迭代和值迭代法的动态规划matlab代码,实现机器人的最优运输 Dynamic-Programming-master/Environment.pdf , 104724 Dynamic-Programming-master/README.md , 506 Dynamic-Programming-master/generalizedPolicyIteration.m , 1970 Dynamic-Programm…...
Git常用命令完全指南:从入门到精通
Git常用命令完全指南:从入门到精通 一、基础配置命令 1. 用户信息配置 # 设置全局用户名 git config --global user.name "你的名字"# 设置全局邮箱 git config --global user.email "你的邮箱example.com"# 查看所有配置 git config --list…...
基于开源AI智能名片链动2 + 1模式S2B2C商城小程序的沉浸式体验营销研究
摘要:在消费市场竞争日益激烈的当下,传统体验营销方式存在诸多局限。本文聚焦开源AI智能名片链动2 1模式S2B2C商城小程序,探讨其在沉浸式体验营销中的应用。通过对比传统品鉴、工厂参观等初级体验方式,分析沉浸式体验的优势与价值…...
渗透实战PortSwigger Labs指南:自定义标签XSS和SVG XSS利用
阻止除自定义标签之外的所有标签 先输入一些标签测试,说是全部标签都被禁了 除了自定义的 自定义<my-tag onmouseoveralert(xss)> <my-tag idx onfocusalert(document.cookie) tabindex1> onfocus 当元素获得焦点时(如通过点击或键盘导航&…...
嵌入式面试常问问题
以下内容面向嵌入式/系统方向的初学者与面试备考者,全面梳理了以下几大板块,并在每个板块末尾列出常见的面试问答思路,帮助你既能夯实基础,又能应对面试挑战。 一、TCP/IP 协议 1.1 TCP/IP 五层模型概述 链路层(Link Layer) 包括网卡驱动、以太网、Wi‑Fi、PPP 等。负责…...
以太网PHY布局布线指南
1. 简介 对于以太网布局布线遵循以下准则很重要,因为这将有助于减少信号发射,最大程度地减少噪声,确保器件作用,最大程度地减少泄漏并提高信号质量。 2. PHY设计准则 2.1 DRC错误检查 首先检查DRC规则是否设置正确,然…...
数据库优化实战指南:提升性能的黄金法则
在现代软件系统中,数据库性能直接影响应用的响应速度和用户体验。面对数据量激增、访问压力增大,数据库性能瓶颈经常成为项目痛点。如何科学有效地优化数据库,提升查询效率和系统稳定性,是每位开发与运维人员必备的技能。 本文结…...
