目录
类加载是指将类的class文件读入内存,并为之创建一个java.lang.Class对象,即当程序使用任何一个类时,系统都会为之创建一个java.lang.Class对象
系统可以在第一次使用某个类时加载该类,也可以采用预加载机制来加载某个类
java.lang.Class类封装一个对象和接口运行时的状态,当加载类时,Class类型的对象将自动创建,Class类没有公共构造方法,其对象是JVM在加载类时通过调用类加载器中的difineClass()方法自动构造的,因此不能显示地实例化一个Class对象
一旦类被载入JVM中,同一个类将不会被再次载入,被载入JVM的类都有一个唯一标识,该标识是该类的全限定域名(包名+类名)
Java程序中获取Class对象中有如下三种方式:
-
- public class ClassDemo {
- public static void main(String[] args) throws ClassNotFoundException {
-
- // 第一种方式
- Class class1 = Class.forName("java.lang.String");
-
- // 第二种方式(提倡)
- Class<String> class2 = String.class;
-
- // 第三种方式
- String string = new String();
- Class class3 = string.getClass();
- }
- }


- package classloader;
-
- import java.lang.reflect.Constructor;
- import java.lang.reflect.Method;
-
- public class ClassDemo {
- public static void main(String[] args) throws ClassNotFoundException, IllegalAccessException, InstantiationException {
-
- // 第一种方式
- Class class1 = Class.forName("java.lang.String");
- System.out.println("①——String类的Class对象:" + class1);
-
- // 第二种方式
- Class<String> class2 = String.class;
- System.out.println("②——String类的Class对象:" + class2);
-
- // 第三种方式
- String string = new String();
- Class class3 = string.getClass();
- System.out.println("③——String类的Class对象:" + class3);
-
- Object object = class1.newInstance();
- System.out.println("String类Class对象的实例:" + object);
- System.out.println("String类的父类:" + class1.getSuperclass());
- System.out.println("String类的所有构造方法:");
- Constructor[] constructors = class2.getDeclaredConstructors();
- for (Constructor constructor : constructors) {
- System.out.println(constructor);
- }
- System.out.println("String类的所有public方法:");
- Method[] methods = class3.getMethods();
- for (Method method : methods) {
- System.out.println(method);
- }
-
- }
- }

当程序主动使用某个类时,若该类未加载入内存,系统将通过加载、连接和初始化三个步骤对类进行初始化
类的加载:由JVM提供的加载器完成,除此之外,也可以通过继承ClassLoader类来创建自定义加载器
加载类来源:
类的连接:将类的二进制数据加入到JRE中
阶段:
类的初始化:对类变量进行初始化
步骤:
类加载器:负责将磁盘或网络上的.class文件加载到内存中,并为之生成对应的java.lang.Class对象

java.lang.ClassLoader是一个抽象类,通过继承该类来实现自定义的用户类加载器,以扩展JVM动态加载类的方式
实现自定义的类加载器,可以通过重写ClassLoader类的以下两种方法:
常用方法:

- package classloader;
-
- import java.io.File;
- import java.io.FileInputStream;
- import java.io.IOException;
-
- class Person{
- public void test(){
- System.out.println("这是Person类的一个方法");
- }
- }
-
-
- class MyClassLoader extends ClassLoader{
-
- // 1.根据指定类名称查找类(重写findClass()方法)
- public Class findClass(String className){
-
- // 2.将返回的字节码数据存储字节数组
- byte[] data = loadClassData(className);
-
- // 3..将二进制数据转为Class对象且返回
- return this.defineClass(className,data,0,data.length);
- }
-
- public byte[] loadClassData(String className) {
-
- // 1.获取当前类Class对象的路径
- String path = this.getClass().getResource("/").getPath();
-
- // 2.从索引为1开始截取路径字符串
- path = path.substring(1);
-
- // 3.替换全限定域名(包名+类名)的分隔符
- className = className.replace(".","/");
-
-
- // 4.获取class文件绝对路径
- File classFile = new File(path + className + ".class");
- long len = classFile.length();
- byte[] classdata = new byte[(int)len];
-
- // 5.读取class文件的全部二进制数据
- FileInputStream fileInputStream = null;
- int res = 0;
- try {
-
- fileInputStream = new FileInputStream(classFile);
- res = fileInputStream.read(classdata);
- } catch (IOException e) {
- e.printStackTrace();
- }
-
- // 6.判断读取的字节数是否与绝对路径中的class文件一致
- if(res != len){
- return null;
- }
- else{
- return classdata;
- }
- }
- }
-
-
- public class ClassLoaderDemo {
- public static void main(String[] args) throws ClassNotFoundException, IllegalAccessException, InstantiationException {
-
- // 1.创建一个类加载器
- MyClassLoader myClassLoader = new MyClassLoader();
-
- // 2.加载Person类(包名+类名),获得Person类的Class对象
- Class cls = myClassLoader.loadClass("classloader.Person");
-
- // 3.获取Class对象的实例
- Person person = (Person)cls.newInstance();
- person.test();
- }
- }
