反射-理解Class类并获取Class实例

反射-理解Class类并获取Class实例

一、class 类介绍

在 Object 类中定义了以下的方法,此方法将被所有子类继承

public final native Class<?> getClass();

getClass 方法返回值的类型是一个 Class类,此类是Java反射的源头,实际上所谓反射从程序的运行结果来看也很好理解,即:可以通过对象反射求出类的名称。

在这里插入图片描述

一个普通类 是 一个Class类的实例

对象的反射类似照镜子,可以得到类的信息。包括:属性、方法和构造器、实现了哪些接口。对于类而言,JRE都为每个类保留一个不变的 Class 类型的对象。一个 Class 对象包含了特定某个结构( class/interface/enum/annotation/ primitive type/void/[])的有关信息。

  • Class 本身也是一个类;
  • Class 对象只能由系统建立;
  • 一个加载的类在JVM中只会有一个 Class实例;
  • 一个 Class对象 对应的是一个加载到JVM中的 一个class文件;
  • 每个类的实例都会记得自己是由哪个 Class实例 所生成;
  • 通过 Class实例 可以完整地得到一个类中的所有被加载的结构;
  • Class类 是 Reflection 的根源,针对任何你想动态加载、运行的类,唯有先获得相应的Class对象。

二、Class类的常用方法

方法名 功能说明
static Class forName (String name) 返回指定 类名 的 Class对象
Object newInstance () 调用 缺省构造函数,返回 Class对象 的一个实例
getName () 返回此 Class对象 所表示的实体(类,接口,数组类或void)的名称。
Class getSuperClass () 返回 当前Class对象 的 父类的Class对象
Class[] getinterfaces () 获取当前 Class对象的接口
ClassLoader getclassLoader () 返回该类的类加载器
Constructor getConstructors () 返回一个包含某些 Constructor对象 的数组
Method getMothed (String name, Class<?>... parameterTypes) 返回一个 Method对象,此对象的形参类型为paramType
Field[] getDeclaredFields () 返回Field对象的一个数组

三、获取Class类的实例

  1. 已知具体的类,通过类的 class属性 获取,该方法最为安全可靠,程序性能最高。
Class clazz = Person.class;
  1. 已知某个类的实例,调用该实例的 getClass() 方法获取Class对象。
Class clazz = person.getClass();
  1. 已知一个类的全类名,且该类在类路径下,可通过 Class类的静态方法 forName(获取,可能抛出 ClassNotFound Exception。
Class clazz = Class.forName("com.gcbeen.Student");
  1. 内置基本数据类型可以直接用类名.Type。
Class clazz = Integer.TYPE;
  1. 还可以利用 Classloader。

完整示例

package com.gcbeen.reflection;
// 测试class类的创建方式有哪些
public class TestCreateClass {

    public static void main(String[] args) throws ClassNotFoundException {
        Person person = new Student();
        System.out.println("这个人是:" + person);

        // 方式一:通过对象查询
        Class c1 = person.getClass();
        System.out.println(c1.hashCode());

        // 方式二:forName获得
        Class c2 = Class.forName("com.gcbeen.reflection.Student");
        System.out.println(c2.hashCode());

        // 方式三:通过类名.class获得
        Class c3 = Student.class;
        System.out.println(c3.hashCode());

        // 方式四:基本类型的包装类都有一个Type
        Class c4 = Integer.TYPE;
        System.out.println(c4);

        // 获得父类类型
        Class c5 = c1.getSuperclass();
        System.out.println(c5);

    }
}

class Person {
    String name;

    public Person() {
    }

    public Person(String name) {
        this.name = name;
    }

    @Override
    public String toString() {
        return "Person{" +
                "name='" + name + '\'' +
                '}';
    }
}

class Student extends Person {
    public Student() {
        this.name = "学生";
    }
}

class Teacher extends Person {
    public Teacher() {
        this.name = "老师";
    }

}

// 这个人是:Person{name='学生'}
// 21685669
// 21685669
// 21685669
// int
// class com.gcbeen.reflection.Person

四、哪些类型可以有Class对象

  1. class:外部类,成员类(成员内部类,静态内部类),局部内部类,匿名内部类。

  2. interface:接口

  3. []:数组

  4. enum:枚举

  5. annotation:注解@interface

  6. primitive type:基本数据类型

  7. void

完整示例

package com.gcbeen.reflection;

import java.lang.annotation.ElementType;

// 所有类型的 Class
public class TestAllTypeClass {

    public static void main(String[] args) {
        Class c1 = Object.class;    // 类
        Class c2 = Comparable.class;    // 接口
        Class c3 = String[].class; // 一维数组
        Class c4 = int[][].class; // 二维数组
        Class c5 = Override.class; // 注解
        Class c6 = ElementType.class; // 枚举类型
        Class c7 = Integer.class; // 基本数据类型
        Class c8 = void.class; // void
        Class c9 = Class.class; // Class

        System.out.println("类: " + c1);
        System.out.println("接口: " + c2);
        System.out.println("一维数组: " + c3);
        System.out.println("二维数组: " + c4);
        System.out.println("注解: " + c5);
        System.out.println("枚举类型: " + c6);
        System.out.println("基本数据类型: " + c7);
        System.out.println("void: " + c8);
        System.out.println("Class: " + c9);

        // 只要元素类型与维度一样,就是同一个Class
        int[] a = new int[10];
        int[] b = new int[100];
        System.out.println(a.getClass().hashCode());
        System.out.println(b.getClass().hashCode());

    }

}

// 类: class java.lang.Object
// 接口: interface java.lang.Comparable
// 一维数组: class [Ljava.lang.String;
// 二维数组: class [[I
// 注解: interface java.lang.Override
// 枚举类型: class java.lang.annotation.ElementType
// 基本数据类型: class java.lang.Integer
// void: void
// Class: class java.lang.Class
// 21685669
// 21685669
posted @ 2022-09-30 10:37  gcbeen  阅读(79)  评论(0)    收藏  举报