Java开发:JVM篇-类加载内存分析

news2024/11/19 2:48:47

一、类加载器

A、类加载器的作用

将class文件字节码内容加载到内存中,并将这些静态数据结构转换成方法区的运行时数据结构,然后在堆中生成一个代表这个类的java.lang.Class对象,作为方法区中类数据的访问入口。

B、类缓存

标准的JavaSE类加载器可以按要求查找类,但一旦某个类被加载到类加载器中,它将维持(缓存)加载一段时间。不过JVM垃圾回收机制可以回收这些Class对象。
示意图
类加载器的作用002

示例代码:


public class Lesson007 {
    public static void main(String[] args) {
        try {
            //系统类加载器
            ClassLoader sClassLoader = ClassLoader.getSystemClassLoader();
            System.out.println("系统类加载器--->" + sClassLoader);

            //扩展类加载器
            ClassLoader eClassLoader = sClassLoader.getParent();
            System.out.println("扩展类加载器--->" + eClassLoader);

            //根加载器(c/c++)
            ClassLoader rClassLoader = eClassLoader.getParent();
            System.out.println("根加载器--->" + rClassLoader);

            //测试当前类是哪个类加载器
            ClassLoader cClassLoader = Class.forName("com.baidu.baike.lesson.annotations_reflection.Lesson007").getClassLoader();
            System.out.println("当前类加载器--->" + cClassLoader);

            //JDK类使用的哪个加载器
            ClassLoader oClassLoader = Class.forName("java.lang.Object").getClassLoader();
            System.out.println("JDK中的类使用的加载器--->" + oClassLoader);

            //如何获得系统类加载器可以加载的路径
            System.out.println("系统类加载器可以加载的路径--->" + System.getProperty("java.class.path"));
        } catch (ClassNotFoundException e) {
            throw new RuntimeException(e);
        }
    }
}

运行结果:
系统类加载器--->sun.misc.Launcher$AppClassLoader@18b4aac2
扩展类加载器--->sun.misc.Launcher$ExtClassLoader@74a14482
根加载器--->null
当前类加载器--->sun.misc.Launcher$AppClassLoader@18b4aac2
JDK中的类使用的加载器--->null
系统类加载器可以加载的路径--->
C:\Users\86157\.jdks\corretto-1.8.0_292\jre\lib\charsets.jar;
C:\Users\86157\.jdks\corretto-1.8.0_292\jre\lib\ext\access-bridge-64.jar;
C:\Users\86157\.jdks\corretto-1.8.0_292\jre\lib\ext\cldrdata.jar;
C:\Users\86157\.jdks\corretto-1.8.0_292\jre\lib\ext\dnsns.jar;
C:\Users\86157\.jdks\corretto-1.8.0_292\jre\lib\ext\jaccess.jar;
C:\Users\86157\.jdks\corretto-1.8.0_292\jre\lib\ext\jfxrt.jar;
C:\Users\86157\.jdks\corretto-1.8.0_292\jre\lib\ext\localedata.jar;
C:\Users\86157\.jdks\corretto-1.8.0_292\jre\lib\ext\nashorn.jar;
C:\Users\86157\.jdks\corretto-1.8.0_292\jre\lib\ext\sunec.jar;
C:\Users\86157\.jdks\corretto-1.8.0_292\jre\lib\ext\sunjce_provider.jar;
C:\Users\86157\.jdks\corretto-1.8.0_292\jre\lib\ext\sunmscapi.jar;
C:\Users\86157\.jdks\corretto-1.8.0_292\jre\lib\ext\sunpkcs11.jar;
C:\Users\86157\.jdks\corretto-1.8.0_292\jre\lib\ext\zipfs.jar;
C:\Users\86157\.jdks\corretto-1.8.0_292\jre\lib\jce.jar;
C:\Users\86157\.jdks\corretto-1.8.0_292\jre\lib\jfr.jar;
C:\Users\86157\.jdks\corretto-1.8.0_292\jre\lib\jfxswt.jar;
C:\Users\86157\.jdks\corretto-1.8.0_292\jre\lib\jsse.jar;
C:\Users\86157\.jdks\corretto-1.8.0_292\jre\lib\management-agent.jar;
C:\Users\86157\.jdks\corretto-1.8.0_292\jre\lib\resources.jar;
C:\Users\86157\.jdks\corretto-1.8.0_292\jre\lib\rt.jar;
D:\code\JavaProject\MyProject\MyProject\out\production\MyProject;
D:\code\JavaProject\MyProject\MyProject\src\lib\commons-io-2.10.0.jar;
D:\software\IDEA\IntelliJ IDEA 2022.2.3\IntelliJ IDEA Community Edition 2022.2.3\lib\idea_rt.jar

二、分析类初始化

什么时候会发生类初始化?

A、类的主动引用(一定会发生类的初始化)

  • 当虚拟机启动,先初始化main方法所在的类
  • new 一个类的对象
  • 调用类的静态成员(除了final常量)和静态方法
  • 使用java.lang.reflect包的方法对类进行反射调用
  • 当初始化一个类,如果其父类没有被初始化,则会先初始化它的父类

B、类的被动引用(不会发生类的初始化)

  • 当访问一个静态域时,只有真正声明这个域的类才会被初始化。如:当通过子类引用父类的静态变量,不会导致子类初始化。
  • 通过数组定义类引用,不会触发此类的初始化
  • 引用常量不会触发此类的初始化(常量在链接阶段就存入了调用类的常量池中了)

public class Lesson006 {
    public static void main(String[] args) {
//        System.out.println("调用Son类的M值为:" + Son.C_M);
//        System.out.println("调用Son类的c_m+" + Son.c_m);

        //主动引用
//        Son son = new Son();


        //反射也会产生主动引用
//        try {
//            Class.forName("com.baidu.baike.lesson.annotations_reflection.Son");
//        } catch (ClassNotFoundException e) {
//            throw new RuntimeException(e);
//        }

        //不会产生类的引用方法
//        System.out.println("调用父类的静态参数f_b = " + Son.f_b);
//        Son[] sons = new Son[5];
//        System.out.println("调用子类的静态常量C_M = " + Son.C_M);
    }
}

class Father {
    static int f_b = 2;

    static {
        System.out.println("父类的静态方法被加载");
    }
}


class Son extends Father {
    static {
        System.out.println("子类的静态方法被加载");
        c_m = 300;
    }

    static int c_m = 100;
    static final int C_M = 1;
    String C_CLASS_NAME = Son.class.getSimpleName();

    public Son() {
        System.out.println("子类的构造函数被加载--->" + C_CLASS_NAME);
    }
}

三、类加载内存分析

类加载内存分析001

A、加载

将class文件字节码内容加载到内存中,并将这些静态数据转换成方法区的运行时数据结构,然后生成一个代表这个类的java.lang.Class对象

B、链接

将Java类的二进制代码合并到JVM的运行状态之中的过程。

  • 验证:确保加载的类信息符合JVM规范,没有安全方面的问题。
  • 准备:正式为类变量(static)分配内存设置类变量默认初始值的阶段,这些内存都将在方法区中进行分配。
  • 解析:虚拟机常量池内的符号引用(常量名)替换为直接引用(地址)的过程。

C、初始化

  • 执行类构造器**方法的过程**。类构造器()方法是编译器自动收集类中所有类变量的赋值动作静态代码块中的语句合并产生的。(类构造器是构造类信息的,不是构造该类对象的构造器)。
  • 当初始化一个类的时候,如果发现其父类还没有进行初始化,则需要先触发其父类的初始化
  • 虚拟机会保证一个类的()方法在多线程环境中被正确加锁和同步。

D、示例

1、示例代码

示例:
public class Lesson005 {
    public static void main(String[] args) {
        System.out.println("获取A类的m值为="+A.m);
    }
}

class A {
    static {
        System.out.println("A类静态代码初始化");
        m = 300;
    }

    static int m = 100;

    public A() {
        System.out.println("A类无参构造初始化");
    }
}

运行结果:
A类静态代码初始化
获取A类的m值为=100

2、运行逻辑分析

a、加载到内存,会产生一个类对应Class对象
b、链接,链接结束后 m=0
c、初始化
(){
System.out.println(“A类静态代码初始化”);
m = 300;
m = 100;
}
d、最终 m=100

3、运行分析图

上述代码运行分析图:
在这里插入class加载运行分析图图片描述

end

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

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

相关文章

收到公安部门的致谢信,顶象业务安全“反诈”再接再厉

12月1日,《反电信网络诈骗法》正式施行,为反电信网络诈骗工作提供有力法律支撑。自2021年以来,顶象业务安全情报通过多渠道的风险数据以及多维度深度挖掘与分析,助力对电信诈骗分子的精准防控,已为多个监管机构的反电信…

小程序开发--- 03组件

小程序中组件的分类主要有以下9类: 常用的视图容器类是组件有: 1.view : 这是普通的视图区域,类似于HTML中的div,是一个块级元素,常用来实现页面的布局效果 2. scroll-view: 是一个可以上下或者左右滚动的区域&#x…

数据结构—set集合

文章目录一、HashSet集合1.HashSet集合的特点2.HashSet常用方法二、LinkedHashSet集合LinkedHashSet集合的特点三、TreeSet集合1.TreeSet集合的特点2.TreeSet的基本使用四、HashSet、LinkedHashSet、TreeSet的使用场景HashSet:LinkedHashSet:TreeSet:五、list和set集…

flutter 自定义加载中间页 loading 菊花组件的封装

flutter 自定义加载中间页前言LoadingStateWidget 封装思路总结前言 在日常移动开发中,很多时候需要我们添加一个加载中间页,即加载中,加载失败,加载重试,加载完成等功能,这样可以避免在无网或者弱网情况下…

《痞子衡嵌入式半月刊》 第 64 期

痞子衡嵌入式半月刊: 第 64 期 这里分享嵌入式领域有用有趣的项目/工具以及一些热点新闻,农历年分二十四节气,希望在每个交节之日准时发布一期。 本期刊是开源项目(GitHub: JayHeng/pzh-mcu-bi-weekly),欢…

【GD32F427开发板试用】IAR 环境移植freertos

本篇文章来自极术社区与兆易创新组织的GD32F427开发板评测活动,更多开发板试用活动请关注极术社区网站。作者:andeyqi freertos移植适配 社区之前已经有同学移植适配freertos,在GD32F427上跑了起来,之前的帖子是在MDK环境下适配的…

FITC-PEG-FA,Folic acid-PEG-Fluorescein,叶酸PEG荧光素

●外观以及性质: FITC-PEG-FA是一种绿色荧光叶酸-PEG衍生物,其激发/发射波长为~490 nm/~520 nm。荧光素标记的PEG叶酸是一种线性分子,叶酸连接到PEG链的一端,荧光素染料连接到另一个PEG端。叶酸也被称为维生素M、维生素B9或叶酸。…

什么?Coolbpf 不仅可以远程编译,还可以发现网络抖动!

近日,在 2022 云栖大会龙蜥峰会 eBPF & Linux 稳定性专场上,来自 eBPF 技术探索 SIG Maintainer 的毛文安分享了《Coolbpf 的应用实践》技术演讲,以下为本次演讲内容: 一、为什么要支持可移植? 随着 BPF 技术的发…

EL表达式 过滤器 监听器

这里写目录标题1 EL 表达式1.1 EL 表达式介绍1.2 EL 表达式快速入门1.3 EL 表达式获取数据1.4 EL 表达式注意事项1.5 EL表达式运算符1.6 EL 表达式使用细节1.7 EL 表达式隐式对象2 JSTL2.1 JSTL 介绍2.2 JSTL 核心标签库2.3 JSTL基本使用3 Filter3.1 过滤器介绍3.2 Filter 介绍…

很全很详细的GUI编程

为什么要学习? 写出自己想要的小工具。工作可能需要维护到swing界面。了解MVC架构,了解监听。 组件:窗口、弹窗、面板、文本框、列表框、按钮、图片、监听事件、鼠标、键盘事件、破解工具。 一、简介 GUI的核心技术:Swing AWT 因…

[附源码]计算机毕业设计基于springboot和vue的茶文化交流平台的设计与实现

项目运行 环境配置: Jdk1.8 Tomcat7.0 Mysql HBuilderX(Webstorm也行) Eclispe(IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持)。 项目技术: SSM mybatis Maven Vue 等等组成,B/S模式 M…

路由规则——MVC控制器

MVC的路由规则配置方式比较多,咱们用得最多的是两种: A、全局规则。就是我们熟悉的”{controller}/{action}“。 app.MapControllerRoute(name: "bug",pattern: "{controller}/{action}"); app.MapControllerRoute(name: "八阿哥",pattern: &…

定积分的概念及可积条件

微积分是高等数学的核心,包含微分和积分。前面几篇我们介绍了微分及其逆运算——不定积分(严格来说,不定积分属于微分模块)。传送门:微分与导数 不定积分今天开始,我们进入积分模块。还是老样子&#xff0…

怎么关闭Windows安全启动?

什么是安全启动? 安全启动(英文名为Secure Boot),是装有UEFI固件的较新PC上的一项重要安全功能,它可以保护计算机的启动过程,防止在启动过程中加载未签名的UEFI驱动程序和引导加载程序,以及阻止…

软件测试经典面试题:如何进行支付功能的测试?

非现金支付时代&#xff0c;非现金支付已经成为了生活不可或缺的一部分&#xff0c;我们只需要一台手机便可走遍全国各地&#xff08;前提是支付宝&#xff0c;微信有钱<00>&#xff09;,那么作为测试人员&#xff0c;支付测试也是非常重要的一环&#xff0c;那么下面我就…

Netty03——进阶

目录1.粘包与半包1.1.粘包现象1.2.半包现象1.3.现象分析1.4 解决方案1.4.1.短链接1.4.2.固定长度1.4.3.固定分隔符1.4.4.预设长度2.协议设计与解析2.1.为什么需要协议&#xff1f;2.2.redis 协议举例2.3.http 协议举例2.4.自定义协议要素2.5.编解码器3.聊天室案例3.1.聊天室业务…

Locust学习记录2-locust用户类学习【HttpUser,wait_time,weight和fixed_count属性】

HttpUser 每个模拟用户定义的类&#xff0c;都必须继承HttpUser&#xff0c;目的时为每个用户提供一个client属性&#xff0c;该属性是一个实例HttpSession&#xff0c;可用于向我们要进行负载测试的目标系统发出HTTP请求 当测试开始时&#xff0c;locust将为它模拟的每个用户…

Java内存模型

TOC Java内存模型 Java内存模型&#xff08;Java Memory Model&#xff0c;JMM&#xff09; 是《Java虚拟机规范》中定义的一种用来屏蔽各种硬件和操作系统的内存访问差异&#xff0c;以实现让Java程序在各种平台下都能达到一致性的内存访问效果的一种内存访问模型。从JDK&…

不要再稀里糊涂的使用反射了,四万字带你搞定Java反射(JDK8)

文章目录前言Java反射体系概述类型信息&#xff08;Type&#xff09;GenericDeclarationAnnotatedType成员信息&#xff08;Member&#xff09;属性信息&#xff08;Field&#xff09;构造函数信息&#xff08;Constructor&#xff09;方法信息&#xff08;Method&#xff09;修…

毕业设计-基于机器视觉的指针式仪表智能识别

目录 前言 课题背景和意义 实现技术思路 实现效果图样例 前言 &#x1f4c5;大四是整个大学期间最忙碌的时光,一边要忙着备考或实习为毕业后面临的就业升学做准备,一边要为毕业设计耗费大量精力。近几年各个学校要求的毕设项目越来越难,有不少课题是研究生级别难度的,对本科…