初涉JVM

JVM

字节码、类的生命周期、内存区域、垃圾回收

  • JVM主要功能:

    • 解释运行(翻译字节码
    • 内存管理(GC
    • 即使编译(Just - In - Time, JIT
      • 短时间内常使用到的字节码翻译成机器码存储在内存中,达到减少解释的次数,性能提升,空间换时间
      • 在这里插入图片描述
  • JVM的组成:

    • 字节码文件

      • 组成;

      • 在这里插入图片描述

      • 基本信息:

        • 文件不是通过文件拓展名来确定文件类型的
        • 软件是通过文件的头几个字节去校验文件的类型
        • Java字节码文件中,将文件头称为magic魔数(CAFEBABE)
        • 主副版本号指的是编译字节码文件的JDK版本号
        • 版本号的作用就是判断当前字节码的版本和运行的JDK是否兼容
        • 主版本号 — 44 就是JDK的数字
        • 版本不兼容的解决方案:
        • 升级JDK(容易引发其他问题,需要大量测试)
        • 更换依赖的版本号,以满足JDK的要求(建议采用)
      • 常量池:

      • 作用:避免相同的内容重复定义,节省空间

      • 方法:

        • 操作数栈是临时存放数据的地方
        • 局部变量表是存放方法中的局部变量的位置

在这里插入图片描述
在这里插入图片描述

  • 类的生命周期:

    • 加载 --> 连接(验证–>准备–>解析)–>初始化–>使用–>卸载

    • 加载

      • 类加载器根据类的全限定名通过不同的渠道以二进制流的方式获取字节码信息。

      • 程序员可以使用java代码拓展的不同的渠道:

        1. 本地文件:磁盘上的字节码文件
        2. 动态代理技术:查询运行时使用动态代理生成
        3. 通过网络传输的类:早期Applet技术使用
      • 类加载器在加载完类之后,JVM会将字节码中的信息保存到内存的方法区中,生成一个InstanceKlass,保存类的所有信息

        • 这个InstanceKlass是用CPP编写的,不能用Java操作
      • 同时,JVM还会在堆中生成一份与方法去中数据类似的java.lang.Class对象

        • 这个堆中对象的信息是少于InstanceKlass的,只有需要用到的信息

        • 作用是在Java代码中去获取类的信息以及存储静态字段的数据

      • 方法区和堆中的这两个对象是通过引用相互关联的

      • 对于开发者来说,只需要访问堆中的对象,而不是InstanceKlass,这样JVM就能很好地控制开发者访问数据的范围

    • 连接(验证–>准备–>解析)

      • 验证: 验证内容是否满足《Java虚拟机规范》
      • 准备:给静态变量赋初值(默认值)
        • 如果是final修饰就会这一步直接初始化
      • 解析:将常量池中的符号引用替换成指向** **
        • 符号引用就是字节码文件中使用编号来访问常量池中的内容
    • 初始化:

      • 执行静态代码块中的代码,为静态变量赋值
      • 执行字节码文件中clinit部分的字节码指令
      • 以下四种方式会导致类的初始化:
        1. 访问一个类的静态变量或静态方法,如果是final修饰过的且等号右边是常量,就不会触发初始化
        2. 调用Class.forName(String clsaaName)
        3. new一个该类的对象
          • 构造代码块{ }优于构造方法先执行
        4. 执行Main方法的当前类
      • clinit指令在特定情况下不会触发:
        1. 无静态代码块且无静态变量赋值语句
        2. 有静态变量声明,无静态变量赋值语句
        3. 静态变量用final修饰
      • 还有一点:
        1. 直接访问父类的静态变量,不会触发子类的初始化
        2. 子类的初始化clinit调用之前,会先调用父类的clinit初始化方法
    • 卸载(在GC中讲到)

  • 类加载器(ClassLoader):

    • classloader是JVM提供给应用程序去实现获取类和接口字节码数据的技术

    • 类加载器的分类:(JDK8及以前)

      • 分为两类,一类是java实现,一类是JVM底层源码实现

      • JVM底层CPP实现

        • Bootstrap启动类加载器:加载java中最核心的类
      • Java实现的

        在这里插入图片描述

        • Extension扩展类加载器:允许扩展Java中比较通用的类
        • Application应用程序加载器:加载应用使用的类
    • 双亲委派机制:

      • 作用

        1. 保证类加载的安全性:避免恶意代码替换掉JDK核心类
        2. 避免重复加载
      • 双亲委派机制指的是:

        • 当一个类加载器收到加载类的任务时,会自底向上查找是否加载过,再由顶向下进行加载(看是否在自己的加载路径中)
        • 在这里插入图片描述
      • 如何去主动加载一个类:

        1. 使用Class.forName方法,使用当前类的类加载器去加载指定的类

          ClassLoader classLoader = Demo1.class.getClassLoader();
          
        2. 获取到类加载器,通过类加载器的loadClass方法指定某个类加载器加载

          Class<?> clazz = classLoader.loadClass("com.xxx.xxx");
          
    • 打破双亲委派机制:

      1. 自定义类加载器

        • 关键就是重写loadClass的findclass方法,将双亲委派机制的代码去除
        • Tomcat就是通过这种方式实现应用之间的类隔离
      2. 线程上下文类加载器

        • JDBC的例子:

          在这里插入图片描述

        • SPI就是使用的线程上下文中保存的类加载器进行类的加载,这个类一般是应用程序类加载器

          ClassLoader cl = Thread.currentThread().getContextClassLoader();
          
        • 这种由启动类加载器加载的类,委派应用程序类加载器去加载类的方式,打破了双亲委派机制

        • 当然JDBC这个案例有不同的看法:

          1. 从SPI角度是打破了双亲委派机制的
          2. 如果从类加载器的角度去看,又是符合双亲委派机制的
      3. OSGi模块化

    • JDK8之后的类加载器:

      • 启动类加载器使用java编写,继承自BuiltinClassLoader实现从模块中找到要加载的字节码资源文件
      • 扩展类加载器被替换成平台类加载器,继承自BuiltinClassLoader
  • 运行时数据区:

    • 定义:JVM在运行java程序过程中管理的内存区域

    • 分类:

      在这里插入图片描述

      1. JVM栈

        • JVM栈存的是java方法,每一个方法的调用使用一个栈帧来保存

        • JVM栈存的栈帧多了是会内存溢出的

        • 栈帧的组成:

          • 局部变量表
            • 保存的内容有:实例方法的this对象,方法的参数,局部变量
            • 栈帧的局部变量表是一个数组,数组中每一个位置称之为槽(slot),浮点数类型占用两个slot,其他类型占用一个slot
            • 为了节省空间,表中的slot是可以复用的,一旦某个局部变量不再生效,当前槽可以复用
          • 操作数栈
            • 执行指令的过程中,用来存放临时数据的一块区域
            • 编译器就确定了栈的最大深度
          • 帧数据
            • 保存了直接引用到内存引用的映射关系(动态链接
            • 保存了方法出口
            • 保存了异常表
      2. 本地方法栈

        • 本地方法栈存的是native本地方法的栈帧
        • 在Hotspot虚拟机中,JVM栈和本地方法栈实现上使用了同一个栈空间
      • 一般程序堆内存是看见最大的一块内存区域。

      • 创建的对象都存在堆上

      • 对象放多了也会内存溢出

      • 堆空间有三个需要的值:

        在这里插入图片描述

        1. used:当前已使用的堆内存
        2. total:JVM已经分配的可用堆内存
        3. max:JVM可以分配的最大堆内存
    • 方法区

      • JDK7及之前,方法区是放在堆里面的永久代空间
      • JDK8及之后,方法区放在元空间中,元空间位于OS维护的直接内存中
      • 方法区存放了三部分内容:
        1. 类的元信息:类的基本信息(InstanceKlass对象)
        2. 运行时常量池:字节码文件中的常量池内容
        3. 字符串常量池(放在堆里面):字符串常量
  • 直接内存;

    • 并不属于java运行时的内存区域
    • 引入了NIO机制,直接使用直接内存
    • 主要解决了以下两个问题:
      1. java堆中的对象如果还不再使用要回收,回收时会影响对象的创建和使用
      2. IO操作比如读文件,要先把文件读入内存再把数据复制到堆中。现在直接放入直接内存,同时堆上维护直接内存的引用,减少数据复制的开销
  • GC:

    • PC,JVM栈,本地方法栈这三个是线程不共享的部分,都是伴随着线程的创建而创建,线程的销毁而销毁。而方法的栈帧在执行完方法后就会弹出栈并释放掉对应内存

    • 方法区的回收:

      • 方法区主要就是回收不再使用的
      • 判定一个类是否可以被回收的三个条件:
        1. 该类的所有实例对象都被回收,在堆中没有其实例对象及子类对象
        2. 加载该类的类加载器已经被回收
        3. 该类的java.lang.Class对象没有在任何地方被引用
      • 手动触发回收:System.gc();
    • 堆回收:

      • java对象的是否能被回收,是根据对象是否被引用来决定的

      • 常见的判断方法:

        1. 引用计数法
          • 引用计数法为每个对象维护一个引用计数器,初始值为0,当对象被引用时加1,取消引用减1
          • 优点是实现简单
          • 缺点:
            1. 加一减一都需要维护计数器,对系统系统有一定影响
            2. 存在循环引用问题,对象无法回收
        2. 可达性分析算法
          • 可达性分析将对象分为两类:垃圾回收的根对象和普通对象,对象之间存在引用关系
          • GC Root 对象:
            • 线程Thread 对象
            • 系统类加载器加载的java.lang.Class对象
            • 监视器对象,用来保存同步锁synmchornized关键字持有的对象
            • 本地方法调用使用的全局对象
      • 五种对象引用

        1. 强引用 :可达性算法中描述的对象引用就是强引用

        2. 软引用:

          • 如果一个对象只有软引用关联到它,当程序内存不足时,就会将软引用中的数据进行回收

          • 软引用常用于缓存中

          • SoftReference类实现软引用

          • 把对象包装进软引用对象中:

            new SoftReference<对象类型> (对象)
            
        3. 弱引用:

          • 机制与软引用相似,区别是,GC的时候,不管内存够不够都会直接被回收
          • WeakReference类实现弱引用
          • 弱引用主要在ThreadLocal中使用
        4. 虚引用:

        5. 终结器引用:

      • GC算法

        • GC的过程会通过单独的GC线程来完成,不管是用哪种GC算法,都会有部分阶段需要停止所有用户线程,这被称为STW(stop the world)

        • 判断GC算法优秀的三个指标:

          1. 吞吐量
            • 执行用户代码的时间/(执行用户代码的时间 + GC时间)越大越好
          2. 最大暂停时间
            • STW时间的最大值
          3. 堆使用效率
        • 三个指标不可兼得,一般来说:

          堆内存越大,最大暂停时间越长。想要减少最大暂停时间,就会降低吞吐量

        • 不同的GC算法适用不同的场景

        • 标记 - 清除算法 —> 复制算法 —> 标记 - 整理算法 —> 分代GC

        • 分代GC

          • 结合了前几种算法

          • 在这里插入图片描述

          • 分代回收时,创建出的对象,首先会被放在Eden区

          • Eden区装满后,就会触发年轻代的GC,称为Young GC 或 Minor GC

          • Minor GC 会把Eden区和From区中需要回收的对象回收,把没有回收的对象放进to区

          • 然后S0变成to区,S1变成from区

          • 每次Eden区装满后,就会触发 Minor GC,对象的年龄会+1

          • 对象的年龄达到阈值(max = 15)会被晋升到老年代

          • 当老年代空间不足时,就会触发Full GC,对整个堆进行GC

        • 垃圾回收器

          • 垃圾回收器是GC算法的具体实现

          • 由于垃圾回收器也是分为年轻代和老年代,除G1外其他的垃圾回收器必须成对组合进行使用

          • 在这里插入图片描述

          • JDK9后默认的垃圾回收器是G1 垃圾回收器

          • Parallel Scavenge 关注吞吐量,允许用户设置最大暂停时间,但是会减少年轻代可用空间的大小

          • CMS关注暂停时间,但是吞吐量方面会下降

          • G1 垃圾回收器的设计目标就是将上面两种垃圾回收器的优点融合:

            1. 支持巨大的堆空间回收,并有较高的吞吐量’
            2. 支持CPU并行垃圾回收
            3. 允许用户设置最大暂停时间
          • G1的整个堆被划分为多个大小相等的区域,区域不要求是连续的,区的大小是堆空间大小/2048

            在这里插入图片描述

          • G1的GC方式有两种:

            1. Young GC
            2. Mixed GC
          • GC流程:

            1. 先创建的对象放在Eden区。
            2. 当G1判断年轻代区不足(max默认60%),无法分配对象时会执行Young GC
            3. 标记出Eden和Survivor区中的存活对象
            4. 根据配置的最大暂停时间选择某些区将存活对象复制到一个新的Survivor区中(年龄+1),清空这些区
            5. 当某个对象的年龄达到阈值(默认15),会被放到老年代
            6. 部分对象如果大小超过区的一半,会被直接放到老年代,这类老年代被称为Humongous区
            7. 对象的总堆占有率达到阈值(默认45%),会触发Mixed GC。回收部分老年代和全部年轻代以及大对象区。采用复制算法完成
            8. G1对老年代的清理会选择存货度最低的区域来进行回收,这样就可以保证回收效率最高,就是G1(Garbage first)名字的由来
            9. 最后的清理阶段使用复制算法,不会产生内存碎片
            10. 如果清理过程中发现没有足够的空区存放转移的对象,会出现Full GC
              Survivor区中(年龄+1),清空这些区
            11. 当某个对象的年龄达到阈值(默认15),会被放到老年代
            12. 部分对象如果大小超过区的一半,会被直接放到老年代,这类老年代被称为Humongous区
            13. 对象的总堆占有率达到阈值(默认45%),会触发Mixed GC。回收部分老年代和全部年轻代以及大对象区。采用复制算法完成
            14. G1对老年代的清理会选择存货度最低的区域来进行回收,这样就可以保证回收效率最高,就是G1(Garbage first)名字的由来
            15. 最后的清理阶段使用复制算法,不会产生内存碎片
            16. 如果清理过程中发现没有足够的空区存放转移的对象,会出现Full GC

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

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

相关文章

【Gin】智慧架构的巧妙砌筑:Gin框架中控制反转与依赖注入模式的精华解析与应用实战(下)

【Gin】智慧架构的巧妙砌筑&#xff1a;Gin框架中控制反转与依赖注入模式的精华解析与应用实战(下) 大家好 我是寸铁&#x1f44a; 【Gin】智慧架构的巧妙砌筑&#xff1a;Gin框架中控制反转与依赖注入模式的精华解析与应用实战(下)✨ 喜欢的小伙伴可以点点关注 &#x1f49d; …

uboot的mmc partconf命令

文章目录 命令格式参数解释具体命令解释总结 mmc partconf 是一个用于配置 MMC (MultiMediaCard) 分区的 U-Boot 命令。具体来说&#xff0c;这个命令允许你设置或读取 MMC 卡的分区配置参数。让我们详细解释一下 mmc partconf 0 0 1 0 命令的含义。 命令格式 mmc partconf &…

【网络安全】子域名模糊测试实现RCE

未经许可&#xff0c;不得转载。 文章目录 正文总结 正文 在之前测试一个私人项目时&#xff0c;我报告了admin.Target.com上的Auth Bypass漏洞&#xff0c;这将导致SQLI&RCE &#xff0c;该漏洞在报告后仅一天就被修复。 现在重拾该应用程序&#xff0c;对子域进行模糊测…

探索 Blockly:自定义积木实例

3.实例 3.1.基础块 无输入 , 无输出 3.1.1.json var textOneJson {"type": "sql_test_text_one","message0": " one ","colour": 30,"tooltip": 无输入 , 无输出 };javascriptGenerator.forBlock[sql_test_te…

c语言第四天笔记

关于 混合操作&#xff0c;不同计算结果推理 第一种编译结果&#xff1a; int i 5; int sum (i) (i) 6 7 13 第二种编译结果&#xff1a; int i 5; int sum (i) (i) 6 7 7 7 前面的7是因为后面i的变化被影响后&#xff0c;重新赋值 14 第一种编译结果&#xff…

后端解决跨域(Cross-Origin Resource Sharing)(三种方式)

注解CrossOrigin 控制层的类上或者方法上加注解CrossOrigin 实现接口并重写方法 Configuration public class CorsConfig implements WebMvcConfigurer {Overridepublic void addCorsMappings(CorsRegistry registry) {// 设置允许跨域的路径registry.addMapping("/**&qu…

springboot配置文件如何读取pom.xml的值

比如想读取profile.active的值&#xff0c;默认属性为pro 在maven中加入以下插件&#xff1a; <plugin><groupId>org.apache.maven.plugins</groupId><artifactId>maven-resources-plugin</artifactId><version>3.2.0</version>&l…

Servlet详解(超详细)

Servlet详解 文章目录 Servlet详解一、基本概念二、Servlet的使用1、创建Servlet类2、配置Servleta. 使用web.xml配置b. 使用注解配置 3、部署Web应用4、处理HTTP请求和生成响应5、处理表单数据HTML表单Servlet 6、管理会话 三、servlet生命周期1、加载和实例化2、初始化3、 请…

pinia安装及简介

pinia简介 基本特点 轻量级&#xff1a;Pinia相比于传统的Vuex&#xff0c;体积更小&#xff0c;性能更好&#xff0c;只有大约1KB左右。 简化API&#xff1a;Pinia简化了状态管理库的使用方法&#xff0c;抛弃了Vuex中的mutations&#xff0c;只保留了state、getters和actions…

科普文:docker基础概念、软件安装和常用命令

docker基本概念 一 容器的概念 1. 什么是容器&#xff1a;容器是在隔离的环境里面运行的一个进程&#xff0c;这个隔离的环境有自己的系统目录文件&#xff0c;有自己的ip地址&#xff0c;主机名等。也可以说&#xff1a;容器是一种轻量级虚拟化的技术。 2. 容器相对于kvm虚…

基于Golang+Vue3快速搭建的博客系统

WANLI 博客系统 项目介绍 基于vue3和gin框架开发的前后端分离个人博客系统&#xff0c;包含md格式的文本编辑展示&#xff0c;点赞评论收藏&#xff0c;新闻热点&#xff0c;匿名聊天室&#xff0c;文章搜索等功能。 项目在线访问&#xff1a;http://bloggo.chat/ 访客账号…

SMU Summer 2024 Contest Round 7

Bouquet 思路&#xff1a; 总的方案数就是C(n,1)C(n,2) . . . . C(n,n) &#xff1b;然后不符合的方案数为C(n,a)C(n,b); 两者相减就是答案&#xff1b;因为算组合数时&#xff0c;数据非常大&#xff0c;所以要用到lucas定理来计算组合数的大小&#xff1b; 当同余定理用…

C#使用Clipper2进行多边形合并、相交、相减、异或的示例

Clipper2库介绍 开源库介绍&#xff1a; Clipper2在Github上的地址&#xff1a;https://github.com/AngusJohnson/Clipper2 Clipper2库对简单和复杂多边形执行交集&#xff08;Intersection&#xff09;、并集&#xff08;Union&#xff09;、差分&#xff08;Difference&…

Python安装

download 1、下载 后直接安装 2、cmd运行命令 python

kafka详解及应用场景介绍

Kafka架构 Kafka架构&#xff0c;由多个组件组成&#xff0c;如下图所示&#xff1a; 主要会包含&#xff1a;Topic、生产者、消费者、消费组等组件。 服务代理&#xff08;Broker&#xff09; Broker是Kafka集群中的一个节点&#xff0c;每个节点都是一个独立的Kafka服务器…

Oracle集群RAC磁盘管理命令asmcmd的使用

文章目录 ASM磁盘共享简介ASM磁盘共享的优势ASM磁盘组成ASM磁盘共享的应用场景Asmcmd简介Asmcmd的功能Asmcmd的命令Asmcmd的使用注意事项Asmcmd运行模式交互模式运行非交互模式运行ASMCMD命令分类实例管理命令:文件管理命令:磁盘组管理命令:模板管理命令:文件访问管理命令:…

Linuxnat网络配置

&#x1f4d1;打牌 &#xff1a; da pai ge的个人主页 &#x1f324;️个人专栏 &#xff1a; da pai ge的博客专栏 ☁️宝剑锋从磨砺出&#xff0c;梅花香自苦寒来 ☁️运维工程师的职责&#xff1a;监…

通信协议_C#实现CAN通信

CAN协议 CAN&#xff08;Controller Area Network&#xff09;即控制器局域网络。特点&#xff1a; 多主网络&#xff1a;网络上的任何节点都可以主动发送数据&#xff0c;不需要一个固定的主节点。双绞线&#xff1a;使用双绞线作为通信介质&#xff0c;支持较远的通信距离。…

用户登录安全是如何保证的?如何保证用户账号、密码安全?

1.HTTP协议直接传输密码&#xff08;无加密&#xff09; 前端 直接发送HTTP请求&#xff08;无加密&#xff09;&#xff0c;攻击者可直接捕获网络包&#xff0c;看到下面的明文信息 因此&#xff0c;使用HTTP协议传输会直接暴露用户敏感信息。 2.HTTPS协议直接传输密码&…

AFSim 仿真系统--任务处理器入门指南

任务处理器 任务处理器&#xff08;WSF_TASK_PROCESSOR&#xff09;是一个WSF处理器&#xff0c;它提供了检查轨迹管理器中的数据&#xff08;轨迹&#xff09;并对其采取行动的能力。这些行动包括&#xff1a; *向下属分配任务 *激活或停用传感器或干扰器 *开火武器 *操纵平台…