反射(Reflection)是Java提供一种在运行时,动态操作类的方法。
在java中,无论是类(Class),枚举(enum),接口(interface),数组(array),注解(annotation),内置数据类型(Primitive type),void类型都包含一个特殊的实例对象,就是java.lang.Class类实例。
java.lang.Class类没有public的构造函数,java.lang.Class类实例只能有JVM自动产生,每次JVM加载Class的时候,JVM就会为其自动生成一个java.lang.Class类实例。

获取java.lang.Class类实例

在java中,可以有两个方法获取类的Class实例。
1. Object类有个getClass()方法,可以获取类的java.lang.Class实例。
2. 每个类有个public的静态变量class。即使是int,long,void这些内置类型也是有class静态变量的,但数组是个比较特殊的,只能通过getClass()方法获取Class变量。如果对数组变量进行.class操作,则会有Unknown Class编译错误。

public class ClassTest {
    public static void main(String[] args) {
        Class<ClassTest> testClass1 = ClassTest.class;
        System.out.println("Class name:"+testClass1.getName());
        //Class name:com.shuanghu.learn.reflection.ClassTest

        Class<ClassTest> testClass2 = ClassTest.class;
        System.out.println("Class name:"+testClass2.getName());
        //Class name:com.shuanghu.learn.reflection.ClassTest

        int [] arr = new int[2];
        System.out.println("Class name:"+arr.getClass());
        //Class name:class [I
        //arr.class;
        //Unknown Class
    }
}

java.lang.Class的使用

判断类和对象的类型

通过java.lang.Class类可以判断一个类的类型。主要可以判断一个类是否是内部类(member class),局部类(local class),枚举(enum),接口(interface),数组(array),注解(annotation),内置数据类型(Primitive type)等类型。java.lang.Class提供了很多成员函数来支持这些判断。

import java.lang.annotation.ElementType;
import java.lang.annotation.Target;

enum TestEnum{
}

interface TestInterface{

}

@Target(ElementType.METHOD)
@interface TestAnnotation {
    String value() default "";
}

public class ClassTest {
    class InnerClass{
    }

    public static void main(String[] args) {
        class LocalClass{
        }

        System.out.println("是否枚举(enum):" + TestEnum.class.isEnum());
        System.out.println("是否接口(interface):" + TestInterface.class.isInterface());
        System.out.println("是否内置类型(Primitive type):" + int.class.isPrimitive());
        System.out.println("是否是内部类:" + InnerClass.class.isMemberClass());
        System.out.println("是否local类:" + LocalClass.class.isLocalClass());
        System.out.println("是否是注解:" + TestAnnotation.class.isAnnotation());

        int [] intArr = new int[2];
        //数组特殊,只支持getClass()获取Class,不支持.class这种静态变量的方式
        System.out.println("是否数组:" + intArr.getClass().isAnnotation());
    }
}
加载类

通过java.lang.Class类可以加载类。在Java中,只有在需要使用一个类的时候,这个类才会被加载。所以,不是所以定义了的类,都会被JVM加载。

java.lang.Class类提供了两个静态方法

public static Class<?> forName(String name) throws ClassNotFoundException;
public static Class<?> forName(String name, boolean initialize, ClassLoader loader) throws ClassNotFoundException;

java.lang.Class的forName通过传入一个类的全名,将类加载到虚拟机。这两个方法的区别就是,单参数forName方法默认是进行初始化的,会执行类的static区域,其实就是执行了forName(className, true,ClassLoader.getClassLoader(Reflection.getCallerClass()))。三参数的forName方法,可以指定类的加载器,以及是否初始化。

package com.shuanghu.learn.reflection;

class DemoLoad1{
    static {
        System.out.println("This is DemoLoad1 class.");
    }
}

class DemoLoad2{
    static {
        System.out.println("This is DemoLoad2 class.");
    }
}

class DemoLoad3{
    static {
        System.out.println("This is DemoLoad3 class.");
    }
}

public class ClassTest {
    public static void test() throws ClassNotFoundException {
        DemoLoad1 c1 = null;
        DemoLoad2 c2 = null;
        DemoLoad3 c3 = null;

        System.out.println("Begin load class DemoLoad1...");
        Class.forName("com.shuanghu.learn.reflection.DemoLoad1");
        Class.forName("com.shuanghu.learn.reflection.DemoLoad2", false, Thread.currentThread().getContextClassLoader());
        System.out.println("Begin init class DemoLoad2...");
        new DemoLoad2();
    }
    public static void main(String[] args) throws ClassNotFoundException {
        test();
    }
}

执行上面的代码,会有如下结果:

Begin load class DemoLoad1...
This is DemoLoad1 class.
Begin init class DemoLoad2...
This is DemoLoad2 class.

从上面结果可以看出,虽然我们定义了三个类DemoLoad1,DemoLoad2和DemoLoad3,但实际上,只有两个类被JVM加载了,即DemoLoad1和DemoLoad2。从执行结果里看,如果仅仅声明类的变量,不进行实际对象的赋值,JVM是不会真实加载类的,如DemoLoad3。单参数版本的forName函数,在类被加载后,即会立马执行类static区域;三参数版本的forName函数,则可以控制类static区域不执行。

Class.forName最常见的一个应用就是jdbc里,通过Class.forName(driverClass)加载不同的java数据库驱动。

通过java.lang.Class实例,可以创建对象

java中创建对象,最常见的方式就是使用new关键字创建。但new并不是唯一的方式。
java.lang.Class有个newInstance方法,可以生成新的对象。但使用newInstance方法生成对象,必须有默认构造函数,或者public,protected的无参构造函数。如果类只有private构造函数,则用newInstance时,会抛出IllegalAccessException异常;如果类只有有参构造函数,则会抛出InstantiationException异常。

class DemoClass1{
    protected DemoClass1(){
        System.out.println("Demo class1 is constructing");
    }
}

class DemoClass2{
    private DemoClass2(){
        System.out.println("Demo class2 is constructing");
    }
}

class DemoClass3{
    private DemoClass3(String param){
        System.out.println("Demo class3 is constructing.Param:"+param);
    }
}


public class ClassTest {

    public static void main(String[] args) throws IllegalAccessException, InstantiationException {
        DemoClass1.class.newInstance();

        DemoClass2.class.newInstance();
        DemoClass3.class.newInstance();
    }
}

虽然newInstance方法会抛出IllegalAccessException和InstantiationException异常,但不代表,这两种情况下就无法通过class创建对象了,后面会有其他方法解决这个问题。

通过java.lang.Class实例,获取类的更详细信息。

仅仅通过java.lang.Class类是无法描述java类的全部信息的。所以,java的java.lang.reflect包里,提供了更多的类。

类名称

作用

java.lang.reflect.Array

描述java的array数组,可以动态访问和新建数据

java.lang.reflect.Constructor

描述java中类的构造函数

java.lang.reflect.Field

描述java类的成员变量

java.lang.reflect.Method

描述java类的方法

上面这些描述类信息的类,java.lang.Class都提供了相应的获取成员函数。