关于java反射的那点事

前言

之前学习的时候有看到过java反射方面的内容,当时以为觉得理解了那些内容,后来才发现时间一长就遗忘了,等用到的时候就到处翻找之前的资料又重新看了一遍,现在还是把这些东西给写出来,下次找的时候就可以很方便了,帮助自己记忆。

实现

说到反射,我们就不得不提Class类了,我们以前都是定义一个类,如下:

class Person(){
    ///////
}

这里我们需要明白的是,我们所定义的这个类Person其实就是Class类的一个对象,这里可能就糊涂了,明明Person是类,怎么是对象呢?

这里我们看一下官方API给出的关于Class类的说明:

Class 类的实例表示正在运行的 Java 应用程序中的类和接口。枚举是一种类,注释是一种接口。每个数组属于被映射为 Class 对象的一个类,所有具有相同元素类型和维数的数组都共享该 Class 对象。基本的 Java 类型(boolean、byte、char、short、int、long、float 和 double)和关键字 void 也表示为 Class 对象。
Class 没有公共构造方法。Class 对象是在加载类时由 Java 虚拟机以及通过调用类加载器中的 defineClass 方法自动构造的。

说的应该很清楚了,Class类的实例也就是对象是java程序中的类和接口,我们定义了一个类也即是生成了Class类的一个 对象。
因为Class没有公共的构造方法,所以我们该怎么去得到一个Class的对象呢?
原来每一个类都有一个Class的对象代表当前类在内存中的引用,我们只要得到这个Class对象就可以了。而得到Class对象有三种方式,这里我们已Person类为例来得到Person类的Class对象。

  • Class cla = Person.class

  • Person p = new Person();
    Class cla = p.getClass();

  • Class cla = Class.forName(“com.example.model.Person”);

以上三种方式我们都可以得到Class对象,第一种就是直接用Person类调用其静态变量class也即是Class的对象,第二种是通过Person的一个对象调用getClass()方法来得到Class对象,第三种也是最常用的一种,通过Person类的完整类名的字符串调用Class类的静态方法forName来得到Class对象,此时cla对象也就是Person类在内存中的引用,通过cla我们同样可以生成Person类的一个对象,如下所示:

public Person(){
        System.out.println("默认的构造函数被调用");
    }
    public void test1(){
        Class cla = Person.class;
        Person p = null;
        try {
            p = (Person) cla.newInstance();
        } catch (Exception e) {}
    }

test1的运行结果为:
这里写图片描述
我们就可以看到,Person的构造函数被执行了,因为我们调用了Class类的newInstance方法,该方法在API的解释是:
这里写图片描述
所以我们就通过cla对象得到了Person类的一个对象。这也就意味着,你给我一个完整的类名的字符串,我通过Class类就可以得到该类的一个对象,这是什么有用的。

上面我们只是调用Person类的默认构造方法,但是实际我们的构造方法都是有参数的,那么我们该怎么通过Class来调用参数的构造方法来得到Person类的对象?
别急,这里我们就要说另一个类了,也是在反射中十分重要的一个类:Constructor
这里写图片描述
我们可以看到,Constructor提供关于类的单个构造方法的信息,也即是说我们得到了Constructor的对象就得到了类的构造方法,然后通过Constructor来调用构造方法是不是就得到了类的对象呢?
在Class类的方法中,有几个方法是关于Constructor的:

这里写图片描述
这里写图片描述
其中前两个方法得到是public类型的构造方法,后面得到了是所有的构造方法包括private类型在内。

    public Person(String name){
        System.out.println("name = " + name);
    }

    private Person(int age){
        System.out.println("age = " + age);
    }

这里我们再给出两个构造方法,一个带有String类型的参数,另一个是int类型的并且是私有的构造方法。我们可以测试一下:

    public void test2() throws Exception{
        //得到字节码
        Class cla = Class.forName("com.example.model.Person");
        //调用无参的构造方法
        Constructor con1 = cla.getConstructor(null);
        Person p1 = (Person) con1.newInstance(null);

        //调用参数为String类型的构造方法
        Constructor con2 = cla.getConstructor(String.class);
        Person p2 = (Person) con2.newInstance("Tom"); 

        //调用参数为int类型的私有构造方法
        Constructor con3 = cla.getDeclaredConstructor(int.class);
        con3.setAccessible(true);
        Person p3 = (Person) con3.newInstance(18);
    }

运行结果如下:

这里写图片描述

可以看到,我们确实调用了不同的构造方法连同私有的构造方法一起。

反射方法

我们不仅可以反射构造方法,我们同样可以反射实例方法,通过方法名来直接调用方法。下面是我定义的几个方法:

    public void m1(){
        System.out.println("方法m1被调用");
    }

    public void m2(String name){
        System.out.println("方法m2被调用"+" name= "+name);
    }

    private void m3(){
        System.out.println("私有方法m3被调用");
    }

    public static void m4(){
        System.out.println("静态方法m4被调用");
    }

然后通过下面的代码去测试一下:

    public void test3() throws Exception{
        Class cla = Person.class;
        Person p = (Person) cla.newInstance();
        //得到方法m1
        Method m1 = cla.getMethod("m1", null);
        m1.invoke(p, null);

        //得到方法m2
        Method m2 = cla.getMethod("m2", String.class);
        m2.invoke(p, "Tom");

        //得到私有方法m3
        Method m3 = cla.getDeclaredMethod("m3", null);
        m3.setAccessible(true);
        m3.invoke(p, null);

        //得到私有方法m4
        Method m4 = cla.getMethod("m4", null);
        m4.invoke(null, null);
    }

运行结果如下:

这里写图片描述

可以看到所有的方法都成功调用了。通过方法名,我们就可以调用相应的方法。

反射字段

    public String name = "JcMan";   

我们定义一个字段,然后我们通过Class来得到这个字段的值。

    public void test4() throws Exception{
        Class cla = Person.class;
        Person p = (Person) cla.newInstance();
        Field f = cla.getField("name");
        String name = (String) f.get(p);
        System.out.println(name);
    }

运行结果就不用说了。

小结

java的反射机制实在是强大如斯,至于它的应用我也是在某一个开发中用到了一点,但是我却只能用惊叹一次来形容了,精妙的用法,当然一些小的程序是不需要用到了,大的程序自己又暂时不能把握住,所以对反射了解就局限于上面所述,如果有机会,我就把在程序中用到的那部分反射内容介绍给大家。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值