极客-JAVA基础学习总结(三)-Class 类、反射

Class 类

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();    //通过类的实例

使用反射 (reflection) 访问属性/方法

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专栏

猜你喜欢

转载自blog.csdn.net/u012002125/article/details/106821059