1、Class 类是代表类的类。每个Class类的实例,都代表了一个类
2、在java世界里,一切皆对象。从某种意义上来说,java有两种对象:实例对象和Class对象。每个类的运行时的类型信息就是用Class对象表示的。它包含了与类有关的信息。其实我们的实例对象就通过Class对象来创建的。Java使用Class对象执行其RTTI(运行时类型识别,Run-Time Type Identification),多态是基于RTTI实现的。
3、每一个类都有一个Class对象,每当编译一个新类就产生一个Class对象,基本类型 (boolean, byte, char, short, int, long, float, and double)有Class对象,数组有Class对象,就连关键字void也有Class对象(void.class)。Class对象对应着java.lang.Class类,如果说类是对象抽象和集合的话,那么Class类就是对类的抽象和集合
public class ClassOfClassAppMain { public static void main(String... args) throws NoSuchFieldException, NoSuchMethodException { LittleSuperMarket superMarket = new LittleSuperMarket("卖场", "世纪大道", 500, 600, 100); MerchandiseV2 m100 = superMarket.getMerchandiseOf(100); // >> TODO // Object类里的getClass方法,可以得到 Class clazz = ShellColorChangePhone.class; // System.out.println(clazz.getName()); // System.out.println(clazz.getSimpleName()); // TODO 通过一个类的Class实例,可以获取一个类所有的信息,包括成员变量,方法,等 // Field countField = clazz.getField("count"); // Field nameField = clazz.getField("count"); //Field 为成员变量 Field countField = clazz.getField("count"); // >> TODO 变长参数和它的等价形式 // Method equalsMethod = clazz.getMethod("equals", Object.class); //Method 为方法 Method buyMethod = clazz.getMethod("buy", int.class, String.class); Method equalsMethod = clazz.getMethod("equals", Object.class); } }
在运行状态中,对于任意一个类,都能够获取到这个类的所有属性和方法,对于任意一个对象,都能够调用它的任意一个方法和属性(包括私有的方法和属性),这种动态获取的信息以及动态调用对象的方法的功能就称为java语言的反射机制。通俗点讲,通过反射,该类对我们来说是完全透明的,想要获取任何东西都可以。
想要使用反射机制,就必须要先获取到该类的字节码文件对象(.class),通过字节码文件对象,就能够通过该类中的方法获取到我们想要的所有信息(方法,属性,类名,父类名,实现的所有接口等等),每一个类对应着一个字节码文件也就对应着一个Class类型的对象,也就是字节码文件对象。
获取字节码文件对象的三种方式。
1、Class clazz1 = Class.forName("全限定类名"); //通过Class类中的静态方法forName,直接获取到一个类的字节码文件对象,此时该类还是源文件阶段,并没有变为字节码文件。
2、Class clazz2 = Person.class; //当类被加载成.class文件时,此时Person类变成了.class,在获取该字节码文件对象,也就是获取自己, 该类处于字节码阶段。
3、Class clazz3 = p.getClass(); //通过类的实例
public class ReflectionAppMain { public static void main(String... args) throws NoSuchFieldException, IllegalAccessException, NoSuchMethodException, InvocationTargetException { LittleSuperMarket superMarket = new LittleSuperMarket("大卖场", "世纪大道1号", 500, 600, 100); MerchandiseV2 m100 = superMarket.getMerchandiseOf(100); Class clazz = m100.getClass(); // TODO 另一种获得Class实例的方法,直接类名点 Class clazz = MerchandiseV2.class; Class clazz = m100.getClass(); // // Field countField = clazz.getField("count"); // System.out.println("通过反射获取count的值:"+countField.get(m100)); // Method buyMethod = clazz.getMethod("buy", int.class); //invoke表示调用--通过反射调用buy方法 System.out.println(buyMethod.invoke(m100, 10)); //通过反射访问静态Field // Field field = clazz.getField("STATIC_MEMBER"); // System.out.println(field.get(null)); //使用反射得到一个private方法的实例 Method descMethod = clazz.getDeclaredMethod("describe"); //强制设置成可以访问 descMethod.setAccessible(true); //调用describe方法m100表示在哪引用对象上调用 descMethod.invoke(m100); //在其他对象上调用describe方法 descMethod.invoke(superMarket.getMerchandiseOf(0)); descMethod.invoke(superMarket.getMerchandiseOf(10)); //调用静态方法 //Method staticMethod = clazz.getMethod("getNameOf", MerchandiseV2.class); //String str = (String) staticMethod.invoke(null, m100); //System.out.println(str); // 调用一个有参的实例方法 // Method buyMethod = clazz.getMethod("buy", int.class); // buyMethod.invoke(m100, 1); // m100.buy(10); } }
参考:
极客时间-零基础学Java专栏