好久没有写了,本想着接着上次的数据结构写下关于链表的,最近看到了java反射这一块东西,索性先来看看反射,想必大家都知道反射重要性,在后面的框架中基本都是反射,所以说我们先要了解什么是反射,为后面的框架打下基础,不多说了,我们进入正题。
一、类加载机制
在我们要了解什么是反射之前,我们要了解一个概念(类加载机制),这里也不深究,若想深入的学习就去看看JVM这一块,这里我们只说一下类的加载过程即可,当我们创建了一个类时,却不知道JVM帮我们都做了些什么,从一个类的编译(编译成字节码文件)再到加载和运行,这期间都发生了什么,如图所示。
二、 类的加载过程
上图可知,当我们创建一个类时,当它被加载到内存中时发生了七个阶段,只要在初始化之前都属于加载过程,我们来了解下每个部分。
1. 类的加载
> 1.类的加载是指将类的class文件(字节码文件)载入到内存中,此时为我们创建一个java.lang.Class对象,并且我们称它为字节码对象。
2.类的加载过程是由类加载器(classLoader)完成,一般类加载器由JVM提供,我们也可以自定义它但是要继承ClassLoader。
2. 类的连接
> 当类被加载到内存中之后,与之同时会有一个Class对象,接着把类的数据以二进制的形式合并到JRE中。
1.验证 过程:即检测被加载的类是否有正确的类结构。
2.准备过程:负责为类的static变量分配内存,并设置默认值。
3.解析过程:把类的二进制数据中的符号引用替换为直接引用。注:(符号引用是一个字符串,它给出了被引用的内容的名字并且包含关于这个被引用项的信息,这些信息可以是一个类、字段、方法等,这样对于其他类的符号引用必须是给出类的全类名)。
3.类的初始化
> 在此阶段,JVM主要负责对类进行初始化,主要是是static变量的初始化
1. 如果该类还未被加载和连接,则先进行加载并连接该类。
2.如果该类的直接父类还未被初始化,则先初始化其直接父类。
3.如果类中有初始化语句(静态代码块),则依次执行这些初始化语句。
三、反射机制
通过上述,我们大概了解了类的整个完整的加载过程接着我们正式进入我今天的主题,》反射。我们通过两个简单的问题来引出我们的反射。
问题1:
> 1. 我们都知道对象有编译类型和运行类型如:
Object obj = new Date();
编译类型:Object
运行类型: new Date;
需求:通过obj对象来调用Date类中的toLoccaleString方法,即
obj.toLoccaleString();我们可以发现编译竟然报错。该如何解决了?
解决办法:其实很简单,我们只需要将obj强转为Date类型,为啥了,因为obj
真实类本来就是Date类型,然后就可以了。
2.试想一下,假设我们不知道obj的真实类型,此时就无法强制转化,那么问题
来了,又该如何解决了?想必大家都有了想法,我们这里先不说,来看看下一个问题。
问题2:
> 我们都知道,在java界中,有一句这样的话"一切皆对象",那么问题来了,类
这种事物又是啥对象,又应该使用怎么样的类来描述这种对象了?
1.我在这里创建几个类如下:
Date d1 = new Date();
Date d2 = new Date();
Date d3 = new Date();
String s1 = new String("aaaa");
String s2 = new String("123");
String s3 = new String("cccc");
ArrayList list1 = new ArrayList();
ArrayList list2 = new ArrayList();
ArrayList list3 = new ArrayList();
我在这里创建了9个对象,那么谁能够描述这9个对象了,我们来分析:
由上图我们可以回答了问题2,我们最后将将这三个类由java.lang.Class来描述,综述上面两个小问题,我们可以发现,最后都是要得到元数据的过程,其实这就是反射的原理,即在运行时期,动态的去获取一个类中的成员信息(如构造器、方法、字段、内部类和接口)等。注(类一旦加载进内存,就会变成Class对象,即字节码文件),本次我们就先了解java反射,下次我们来深入学习它。