Java反射机制

反射

Java的反射就是指程序在运行期间可以拿到一个对象所有的信息。(为了解决在运行期间,对某个实例一无所知的情况下,如何调用其方法)
class
Method
Field
Constructor

获取class对象的三种方式:
1.在源码阶段,Class.forName(“全类名”):通过静态方法获取class对象,将字节码文件加载进内存,返回class对象
2.字节码文件阶段,类名.class :通过类名的属性class获取;
3.运行阶段,对象.getClass():通过调用实例对象提供的方法
注意:getClass对象在object类中,也就意味着所有子类都继承了这个方法
在这里插入图片描述

类加载

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

class是有JVM执行过程中动态加载的。JVM在第一次读取到一种class类型时,就将其加载进内存。
每加载一种class,JVM就会为其创建一个class类型的实例,并关联起来。

类的生命周期分为七个阶段:加载/验证/准备/解析/初始化/使用/卸载
JVM类加载机制分为五个部分:加载/验证/准备/解析/初始化
在这里插入图片描述加载:主要完成三件事情:1.通过类的全限定名来获取此类的二进制字节流;2.将这个类字节流代表的静态存储结构转为方法区的运行时数据结构;在堆中生成一个代表此类的java.lang.Class对象,作为访问方法区这些数据结构的入口。

查找并加载类的二进制文件(也就是.class文件);方法区:存放类的类信息;堆:class文件对应的实例对象。
这个过程主要就是通过类加载器完成。

校验:此阶段主要确保Class文件的字节流中包含的信息符合当前虚拟机的要求,并且不会危害虚拟机的自身安全;
主要包括:1.文件格式验证:基于字节流的验证;2.元数据的验证:基于方法区的存储结构验证;3.字节码验证:基于方法区的存储结构验证;4.符号引用验证:基于方法区的存储结构验证。

准备:为类变量分配内存,并将其初始化为默认值(此时为默认值,在初始化的时候才会给变量赋值)即在方法区中分配这些变量所使用的内存空间。类变量是static修饰的变量,不包括实例变量,实例变量会在对象实例化的时候随着对象一起分配在Java堆上。

解析:该阶段就是将JVM中常量池内的符号引用替换成直接引用的过程。

初始化:执行类的初始化方法,方法clinit是由编译器自动生成的,为静态变量赋予正确值。
注意:父类定义的静态代码块优先于子类的静态代码块先执行;
子类引用父类的静态字段不会被初始化。

双亲委派机制

在这里插入图片描述双亲委派加载机制:如果一个类加载器接收到了类加载的请求,他首先不会自己去尝试加载这个请求,而是把请求委托给父加载器去完成,依此向上。因此,所有的类加载请求最终都应该被传递到顶层的启动类加载器中。只有当父加载器在他的搜索范围中没有找到所需的类时,即无法完成加载,子类才会尝试自己去加载该类。
使用这种机制的好处是:java类随着他的类加载器一起具备了一种带有优先层级的层级关系。例如java.lang.Object,它存在于rt.jar中,无论哪一个类加载器要加载这个类,最终都是委派给处于模型最顶端的Bootstrap ClassLoader进行加载,因此Object类在程序的各种类加载环境中都是同一个类。相反,如果没有双亲委派模型,而是由各个类加载器自行加载的话,如果用户编写了一个java.lang.Object的同名类并放在ClassPath中,那系统将会出现多个不同oobject类,程序将混乱。但是如果用了双亲委派模型之后,如果开发者尝试编写一个同名的类,可以被编译但是永远不会被加载。将最大的权限交给核心类库,防止API库被随意篡改。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值