先了解下在程序准备运行某个类,但是该类还没被加载到内存中,会经过以下三个步骤:

类的加载(Load)→类的连接(Link)→类的初始化(Initialize)

  • 加载:类经过javac.exe编译的.class字节码文件读入内存(将静态数据转换成堆中方法区的运行时数据结构),并为之创建一个java.lang.Class对象作为方法区中类数据的访问入口(引用的地址),需要访问和使用类数据只能通过这个Class对象;此过程由类的加载器完成;
  • 链接:将java类的二进制代码合并到JVM的运行状态中的过程;
    • 验证:确保加载的类符合JVM规范;
    • 准备:正式为类变量(static)分配内存并设置变量默认初始值(非任何显示赋值),这些内存都在方法区中分配;
    • 解析:虚拟机常量池内的符号引用(常量名)替换为直接引用(地址)的过程
  • 初始化:JVM负责对类进行初始化;
    • 执行类构造器()方法的: 此方法是由编译期自动收集类中所有类变量的赋值动作和静态代码块中的语句合并产生的(类构造器是构造类信息的,并非new对象构造器)
    • 如其父类为进行初始化,则初始化操作从先从父类进行;
    • 虚拟机会保证一个类的()方法在多线程环境中被正确加锁和同步;

类加载器ClassLoader的作用:

除了上面提到的作用,还有一个类缓存机制:一旦某个类被加载到内存中,将位置加载(缓存)一段时间,相当于一个缓存了一个Class对象,无论此类创建多少个实例,都是从这唯一的结构中获取信息;GC也可以回收这些Class对象;

JVM规范定义的类的加载器类型如下:

加载器关系测试:

@Test
public void test1() {
//1.获取一个系统类加载器
ClassLoader systemClassLoader = ClassLoader.getSystemClassLoader();
System.out.println(systemClassLoader);
//2.获取系统类加载器的父类加载器,即扩展类加载器
ClassLoader extensionClassLoader = systemClassLoader.getParent();
System.out.println(extensionClassLoader);
//3.获取扩展类加载器的父类加载器,即引导类加载器
ClassLoader bootstapClassLoader = extensionClassLoader.getParent();
//引导类加载器用于加载java核心库,无法直接获取,故输出null
System.out.println(bootstapClassLoader);
//4.测试当前类由哪个类加载器进行加载
ClassLoader classLoader = null;
try {
classLoader = Class.forName("Reflection.ClassLoaderTest").getClassLoader();
} catch (ClassNotFoundException e) {
e.printStackTrace();
}
System.out.println(classLoader);//结果为系统类加载器
//5.测试JDK提供的Object类由哪个类加载器完成
ClassLoader objClassLoader = null;
try {
objClassLoader = Class.forName("java.lang.Object").getClassLoader();
} catch (ClassNotFoundException e) {
e.printStackTrace();
}
System.out.println(objClassLoader);//结果为null(说明是用的引导类加载器,我们无法获取)
//6.关于类加载器的一个主要方法:getResourceAsStream(String str):获取路径下的指定文件的输入流
InputStream is = null;
is = this.getClass().getClassLoader().getResourceAsStream("Reflection\\test.properties");
System.out.println(is);
//可用于读取配置文件,下面单独拿来测试
}

读取.properties配置文件:

    @Test
public void test2(){
Properties properties = new Properties();//表示一个持久的属性集,可保存在流中或从流中加载
// //1.获取输入流
// //方式一:(此时的文件默认路径在Module下)
// FileInputStream fis = null;
// try {
// fis = new FileInputStream("test.properties");
// } catch (FileNotFoundException e) {
// e.printStackTrace();
// }
//方式二:使用ClassLoader方式(此时的文件默认路径在当前Module的src下)
//获取当前类的Class实例对象-获取类加载器-获取指定指定路径下的文件输入流
InputStream is = this.getClass().getClassLoader().getResourceAsStream("test1.properties"); //2.读取配置文件
try {
//从输入流中读取属性列表(键和元素对)
properties.load(is);
} catch (IOException e) {
e.printStackTrace();
}
//匹配对应key的属性,获取key对应的元素值
String user = properties.getProperty("user");
String password = properties.getProperty("password");
System.out.println("user = " + user + " , password = " + password);
}

Java类的加载过程与ClassLoader的理解及测试的更多相关文章

  1. Java类的加载过程-重点!!

    java类的加载过程有以下几步共同完成: 加载->连接->初始化.连接又分为验证.准备.解析 一个非数组类的加载阶段(加载阶段获取类的二进制字节流的动作)是可控性最强的阶段,这一步我们可以 ...

  2. JAVA - 类的加载过程

    JAVA - 类的加载过程 JVM类加载机制分为五个部分:加载,验证,准备,解析,初始化. 加载 加载是类加载过程中的一个阶段,这个阶段会在内存中生成一个代表这个类的java.lang.Class对象 ...

  3. java 反射,类的加载过程以及Classloader类加载器

    首先自定义一个类Person package reflection; public class Person { private String name; public int age; public ...

  4. java类的加载过程

    1.类的加载顺序 (1)JVM在首次加载类时会对  静态初始化块.静态成员变量. 静态方法进行一次初始化. (2)只有在调用new方法时才会创建类的实例. (3)对象创建过程: 首先执行父类(如果有) ...

  5. Java 类的加载过程(阿里面试题)

    问以下程序打印出什么内容: 问题及解析如下: /** * 加载方法不等于执行方法,初始化变量则会赋值 * 类加载顺序应为 加载静态方法-初始化静态变量-执行静态代码块 * 实例化时 先加载非静态方法- ...

  6. Java类的加载、链接和初始化

    一.Java的类加载机制回顾与总结: 我们知道一个Java类要想运行,必须由jvm将其装载到内存中才能运行,装载的目的就是把Java字节代码转换成JVM中的java.lang.Class类的对象.这样 ...

  7. 别翻了,这篇文章绝对让你深刻理解java类的加载以及ClassLoader源码分析【JVM篇二】

    目录 1.什么是类的加载(类初始化) 2.类的生命周期 3.接口的加载过程 4.解开开篇的面试题 5.理解首次主动使用 6.类加载器 7.关于命名空间 8.JVM类加载机制 9.双亲委派模型 10.C ...

  8. java类从加载、连接到初始化过程

    类加载器 在了解Java的机制之前,需要先了解类在JVM(Java虚拟机)中是如何加载的,这对后面理解java其它机制将有重要作用. 每个类编译后产生一个Class对象,存储在.class文件中,JV ...

  9. jvm系列(一):java类的加载机制

    java类的加载机制 1.什么是类的加载 类的加载指的是将类的.class文件中的二进制数据读入到内存中,将其放在运行时数据区的方法区内,然后在堆区创建一个java.lang.Class对象,用来封装 ...

随机推荐

  1. mongo windows 安装

    下载安装包 一路next 打开cmd 或者 power shell 准备本地目录. D: #进入d盘 md data #创建目录 data cd data #进入目录 data md config # ...

  2. 艾编程coding老师:深入JVM底层原理与性能调优

    1. Java内存模型JMM,内存泄漏及解决方法:2. JVM内存划分:New.Tenured.Perm:3. 垃圾回收算法:Serial算法.并行算法.并发算法:4. JVM性能调优,CPU负载不足 ...

  3. python判断是否是质数

    质数:只能被1和它自身整除 # 获取用户输入的数,判断是否是质数 num = int(input('输入一个任意的大于1的整数:')) i=2 # 创建一个变量,记录是否是质数,默认num是质数 fl ...

  4. 通过模拟Mybatis动态代理生成Mapper代理类,讲解Mybatis核心原理

    本文将通过模拟Mybatis动态代理生成Mapper代理类,讲解Mybatis原理 1.平常我们是如何使用Mapper的 先写一个简单的UserMapper,它包含一个全表查询的方法,代码如下 pub ...

  5. redis订阅发布简单实现

    适用场景 业务流程遇到大量异步操作,并且业务不是很复杂 业务的健壮型要求不高 对即时场景要求不高 原理介绍 redis官网文档:https://redis.io/topics/notification ...

  6. springboot集成restdocs输出接口文档

    1.pom文件新增restdocs <dependency>    <groupId>org.springframework.restdocs</groupId> ...

  7. 编程基础系列--之--浅谈List、Set、Map和泛型(一)——单列集合

    之前在学习Java时对于泛型,集合的理解一直模模糊糊,随着时间的推移,对泛型和集合有了一些浅显的认知,打算写出来巩固一下,也希望各位大佬能指出理解不当之处,万分感谢!!! 在Java语言中,集合分为两 ...

  8. Git The requested URL returned error:403

    一.问题描述 hexo部署的仓库需要换到另一个账号上,于是按照之前创建的步骤,修改了hexo安装目录下的deploy的repo地址,然而hexo s和hexo g没出错,但是hexo d时,会出错. ...

  9. NOI2.5 8783:单词接龙

    描述 单词接龙是一个与我们经常玩的成语接龙相类似的游戏,现在我们已知一组单词,且给定一个开头的字母,要求出以这个字母开头的最长的"龙"(每个单词都最多在"龙"中 ...

  10. 红黑树(依照4阶B树C++实现)

    我在编写红黑树的时候类比这2-3-4树的原理来书写 语言标准:C++11 在Ubuntu 18.04上通过编译和测试 从刚开始只听说过这个概念,到学习,再到编出代码,然后在进行测试,最后完成代码一共花 ...