java_进阶:类的加载和反射

1、 类加载

当某一个程序在运行时,需要使用某一个类,如果该类还未被加载到内存中,则系统会通过类的加载,类的链接,类的初始化三个步骤来完成对类的初始化。

类的加载:

​ 就是指将class文件读入内存,并位置创建一个java.lang.Class对象。

​ 任何类被使用时,系统都会为之建立一个java.lang.Class对象。

类的链接:

​ 验证阶段:用户检验被加载的类是否有完整的结构,并和其他的类协调一致。

​ 准备阶段: 负责为类的变量分配内存,并设置默认初始值

​ 解析阶段: 将类的二进制数据中的符号引用替换为直接引用

类的初始化:

​ 主要就是对类的变量进行初始化

类的初始化步骤:

​ 假如类还未被加载和链接,则程序先加载并链接该类

​ 假如该类的直接父类还未被加载,则先初始化其直接父类

​ 假如类中具有初始化语句,则系统将一次执行这些初始化语句

注意:在自行第二个步骤的时候,系统对直接父类的初始化步骤也遵循初始化步骤1–3

类的初始化时机:

​ 创建类的实例;

​ 调用类的类方法

​ 访问类或者接口的类变量或者为该类变量赋值

​ 使用反射方式来强制创建某个类或者接口对应的Class对象

​ 初始化某个类的子类

​ 直接使用java

​ 命令来运行某个主类

1.2 类加载器

作用:

​ 负责将class文件加载到内存中,并为之生成对应的Class,虽然我们不用过分的关心类的加载机制,但是了解这个机制就能更好的理解程序的运行

jvm的类加载机制:

  • 全盘负责: 就是当一个类加载器负责加载某个Class的时候,该Class所以来的和引用的其他的Class也将有该类加载器负责载入,除非显式的使用另外一个类加载器来载入

  • 父类委托:就是当一个类加载器负责加载某个Class时,先让父类加载器试图加载,只有在父类加载器无法加载该类时,才尝试从自己的类路径中加载该类

  • -缓存机制:保证所有的加载过得Class在再次使用时,都并不会被重新加载,所有加载过得Class都会被缓存。当需要某个Class对象的时候,类加载器先从缓存区中搜索,只有当缓存区不存在时,系统采用读取该类对应的二进制数据,并将其转换为Class对象,存储到缓存区。

1.3 java中的内置类加载器

Bootstrap class Load: 这是jvm的内置类加载器 通常表示为null,并且没有父类

Paltform class load: 平台类加载器 可以看到所有的平台,包括平台类加载器或其祖先定义的JAVASE平台的API

System class load 应用程序类加载器 与平台类加载器不同,系统类加载器加载通常用于定义应用程序类路径和模块路径和jdk特定工具类

类加载器的继承关系:

​ System的父加载器为Platform 而platform的父加载器为Bootstrap

ClassLoad类表示类加载器:

static ClassLoader getSystemClassLoader() 返回用于委派的系统类加载器。
ClassLoader getParent() 返回父类加载器进行委派。
    public static void main(String[] args) {
   
     ClassLoader  loader =    ClassLoader.getSystemClassLoader();
     System.out.println(loader);//AppClassLoader
      ClassLoader loader1 =   loader.getParent();
      System.out.println(loader1);//PlatformClassLoader
      System.out.println(loader1.getParent());//  null   BootstrapClassLoader
    }


2、反射——框架设计的灵魂

反射指的就是在运行时获取一个类的变量和方法信息,然后通过获取到的信息来创建对象,调用方法的一种机制。——将类的各个组成部分封装为其他对象–反射机制

由于这种动态性,可以极大的增强程序的灵活性,程序不用在编译期就完成确定,在运行期仍然可以扩展。、

2.1 获取Class对象的方式

    public static void main(String[] args) throws ClassNotFoundException {
   
        // 调用了的class属性来获取对应的CLass对象
        Class<Student> c1 = Student.class;
        System.out.println(c1);
        Class<Student> c2 = Student.class;
        System.out.println(c1 == c2);//true  一个类的Class对象在内存中 永远只有一份
        System.out.println("------------------------");
        //调用对象的getClass方法
        Student stu = new Student();
        Class<? extends Student> c3 = stu.getClass();
        System.out.println(c1 == c3);
        System.out.println("------------------------");
        // 使用Class类的静态方法 forName(String className)返回与给定字符串名称的类或接口相关联的 类对象。(推荐)
        Class<?> c4 = Class.forName("cn.lanqiao.clazz.Student");
        System.out.println(c1 == c4)
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值