Java反射详解

转载注明出处://www.greatytc.com/p/2f488de72886

简介

Java在编译时候就必须知道所引用的类所在地方,但是在实际编程中,在某些场合,可能需要引用一个并不在编译空间的类,这个时候常规方法就很难实现了。在Java中,Class配合反射能够很好的解决这种场景。Java里面的反射可以帮助我们在运行程序时候加载、使用编译期间完全未知的class,简单来说就是Java可以加载一个运行时候才得知名称的class,获得其完整的构造,并生成实例化对象,对其成员变量赋值,调用其方法等等。

在具体的研发中,通过反射获取类的实例,大大提高系统的灵活性和扩展性,同时由于反射的性能较低,而且它极大的破坏了类的封装性(通过反射获取类的私有方法和属性),在大部分场景下并不适合使用反射,但是在大型的一些框架中,会大范围使用反射来帮助架构完善一些功能。

上面说了一些简介,接下来直接就看看怎么使用反射来帮助我们获取class中的一些属性和方法吧。

说明

反射机制中会用到一些类,在了解反射是如何使用之前,先介绍一下这些类。

说明
Class 在反射中表示内存中的一个Java类,Class可以代表的实例类型包括,类和接口、基本数据类型、数组
Object Java中所有类的超类
Constructor 封装了类的构造函数的属性信息,包括访问权限和动态调用信息
Field 提供类或接口的成员变量属性信息,包括访问权限和动态修改
Method 提供类或接口的方法属性信息,包括访问权限和动态调用信息
Modifier 封装了修饰属性, public、protected、static、final、synchronized、abstract等

注:Class本身就是一个类,Class就是这个类的名称(注意首字母是大写);public class Demo {},这里的class是作为关键字,来表明Demo是一个类

获取属性

通过反射可以获取类的多种属性,并对其进行一些操作,例如获取其中的某个方法并调用。先定义一个基础的类,然后用反射获取这个类的一些信息吧。

接口,仅仅名声了一个eat()方法

public interface IHumanAction {
    void eat();
}

具体的类,注意类各个方法的修饰属性。

public class Human implements IHumanAction {
    private int age;
    public String name;

    public Human() {

    }

    Human(String name) {
        this.name = name;
    }

    protected Human(int age) {
        this.age = age;
    }

    public Human(int age, String name, String sex) {
        this.age = age;
        this.name = name;
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }
    
    protected void eatApple() {
        System.out.println("eat apple");
    }

    void playGame() {

    }

    @Override
    public void eat() {
        System.out.println(this.name + " performs eat");
    }

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

基类或者接口

最基本的,可以通过反射,来获取一个类的基类或者实现的接口,使用getSuperclass()或者该类的基类,使用getInterfaces()来获取该类实现的接口。直接看一下例子。

try {
    Class clz = null;
    clz = Class.forName("com.wang.demo.reflect.Human");
    if(clz != null) {
        Class superClass = clz.getSuperclass();
        System.out.println("该类的父类:");
        System.out.println(superClass.getName());

        System.out.println("该类实现的接口:");
        Class[] interfaces = clz.getInterfaces();
        for (Class clazz : interfaces) {
            System.out.println(clazz.getName());
        }
    }
} catch (ClassNotFoundException e) {
    e.printStackTrace();
}

运行结果如下图。

图-1 获取基类和接口

因为在java中,类是单一继承,而接口可以实现多个,所以getSuperclass()方法返回的是个Class对象,而getInterfaces()返回的是一个Class数组。

构造函数

获取基类和接口的方法比较单一,是直接返回了类的一些基本属性,而在获取构造函数、方法、成员变量属性时候,不同的方法是返回不同的结果。

先看获取构造函数的例子。

try {
    Class clz = null;
    clz = Class.forName("com.wang.demo.reflect.Human");

    if(clz != null) {
        int modify;
        System.out.println("该类的构造函数(getConstructors()):");

        Constructor[] cons = clz.getConstructors();
        for(Constructor constructor : cons) {
            modify = constructor.getModifiers();
            System.out.println(Modifier.toString(modify) + " " + constructor.getName());
        }

        System.out.println("\n该类的构造函数(getDeclaredConstructors()):");
        Constructor[] cons2 = clz.getDeclaredConstructors();
        for(Constructor constructor : cons2) {
            modify = constructor.getModifiers();
            System.out.println(Modifier.toString(modify) + " " + constructor.getName());
        }

        System.out.println("\n根据参数获取构造函数:");
        Constructor cons3 = clz.getDeclaredConstructor(int.class);
        if(cons3 != null) {
            System.out.println(Modifier.toString(cons3.getModifiers()) + " " + cons3.getName());
        }

        System.out.println("\n根据参数获取构造函数:");
        Constructor cons4 = clz.getConstructor(int.class);
        if(cons3 != null) {
            System.out.println(Modifier.toString(cons4.getModifiers()) + " " + cons4.getName());
        }
    }

} catch (ClassNotFoundException e) {
    e.printStackTrace();
} catch (NoSuchMethodException e) {
    e.printStackTrace();
}

运行结果如下图。

图-2 获取构造函数

在我们定义的Human类中,有四个构造函数,两个public属性的,一个default属性,一个protected属性。

从运行结果中可以看见,getConstuctors()得到了两个构造函数,都是public属性的,getDelaredConstructors()得到了四个构造函数,获取了Human类中的所有构造函数,跟构造函数的属性无关。getDelaredConstructor(params)根据构造函数的参数类型,获取了相匹配的构造函数,而同样的参数getConstuctor(params)抛出了异常,这是因为getConstuctor(params)根据参数去匹配所有的public属性的构造函数,而getDelaredConstructor(params)是根据参数去匹配所有的构造函数。

总体来说,四种获取构造函数的方法的区别如下:

  • getConstuctors(),获取的构造函数全部是public属性的。
  • getConstuctor(Class ... params),根据参数,从所有public属性的构造函数中获取相关构造函数
  • getDelaredConstructors(),获取所有的构造函数
  • getDelaredConstructor(Class ... params),根据参数,从所有的构造函数中获取相关构造函数

方法

在Java中,一般通过getMethods()或者getDeclaredMethod()方法来获取类中定义的方法。直接看一下这二个方法执行的例子。

例子代码:

try {
    Class clz = null;
    clz = Class.forName("com.wang.demo.reflect.Human");
    
    if(clz != null) {
        System.out.println("获取该类的方法(getMethods()):");
        Method[] methods1 = clz.getMethods();
        for(Method method : methods1) {
            System.out.println(Modifier.toString(method.getModifiers()) + " " +
                    method.getReturnType() + " " +
                    method.getName());
        }
    
        System.out.println("\n获取该类的方法(getDeclaredMethods())");
        Method[] methods2 = clz.getDeclaredMethods();
        for(Method method : methods2) {
            System.out.println(Modifier.toString(method.getModifiers()) + " " +
                    method.getReturnType() + " " +
                    method.getName());
        }
    }
} catch (ClassNotFoundException e) {
    e.printStackTrace();
}

运行结果:

图-3 获取方法

这两个方法获取的结果差别还是比较大的,可以看见,getMethods()返回的都是类中public属性的方法,不止是本身声明过的public属性的方法,也包括基类中声明的public属性的方法。而getDeclaredMethods()返回的则是在类自身声明的所有方法,包括复写的方法。

事实上,通过反射获取类中的方法还有getMethod(name,params)getDeclaredMethod(name, params),这两个方法主要区别在于通过参数筛选的范围不同。

四种方法的区别:

  • getMethods()返回类中所有的public属性的方法,包括从基类继承的public方法。
  • getDeclaredMethods()返回类本身声明的方法,包括复写的方法,不包括从基类继承的方法
  • getMethod(name,params)根据参数从getMethods()返回的结果中筛选
  • getDeclaredMethod(name, params)根据参数从getDeclaredMethods()返回的结果中筛选

成员变量

成员变量获取和上面类似,主要方法有getFields()getDeclaredFields()getMethod(name, params)getDeclaredMethod(name, params)四种方法。

看一下使用的例子吧。

try {
    Class clz = null;
    clz = Class.forName("com.wang.demo.reflect.Human");

    if(clz != null) {
        System.out.println("获取该类的成员变量(getFields()):");
        Field[] fields1 = clz.getFields();
        for(Field field : fields1) {
            System.out.println(Modifier.toString(field.getModifiers()) + " " + field.getName());
        }

        System.out.println("\n获取该类的成员变量(getDeclaredFields()):");
        Field[] fields2 = clz.getDeclaredFields();
        for(Field field : fields2) {
            System.out.println(Modifier.toString(field.getModifiers()) + " " + field.getName());
        }

    }
} catch (ClassNotFoundException e) {
    e.printStackTrace();
}

运行结果:

图-4 获取成员变量

可以看到getFields()方法只返回了public属性的成员变量,而getDeclaredFields()方法返回了所有的成员变量。至于getMethod(name, params)getDeclaredMethod(name, params)都是在相应的范围获取成员变量。

四个方法的具体区别:

  • getFields()获取类的所有public属性的成员变量
  • getDeclaredFields()获取类的所有成员变量
  • getMethod(name, params)根据参数在getFields()获取的成员变量中进行筛选
  • getDeclaredMethod(name, params)根据参数在getDeclaredFields()获取的成员变量中进行筛选

调用&赋值

上面说明了获取类的一些属性的方法,包括构造函数、方法和成员变量。在获取这个属性之后我们可以调用构造函数实例化对象,调用其中的方法,给成员变量赋值。

直接看代码。

try {
    Class clz = null;
    clz = Class.forName("com.wang.demo.reflect.Human");

    if(clz != null) {
        Constructor constructor = clz.getDeclaredConstructor(int.class);
        Human human = (Human)constructor.newInstance(1);
        System.out.println(human.toString());

        Method method = clz.getMethod("setName", String.class);
        method.invoke(human, "John");
        System.out.println(human.toString());

        Field field = clz.getDeclaredField("age");
        field.setAccessible(true);
        field.set(human, 12);
        System.out.println(human.toString());
    }
} catch (Exception e) {
    e.printStackTrace();
}

运行结果:

图-5 运行结果

代码中,我先利用反射获取构造函数Human(int age),然后实例化一个Human对象,这个时候打印出Human信息,由于在实例化时候设置了age为1,打印出来结果name是没有值得;然后通过反射获取setName(String name)方法,并调用设置name为Jhon,输出时候可以看见age为1,name为Jhon;最后通过反射获取到Humanage成员变量,并将值设置为12,输出时候可以看见age已经改变为12了。

总结

反射的使用非常简单,在一般场景下也很少用到。但是作为Java中的一个比较重要的辅助机制,还是需要了解。由于反射在获取一些属性时候异常情况比较多,一定需要慎重使用,而且反射性能比较差,在普通场景下,并不建议大量使用反射。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,372评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,368评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,415评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,157评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,171评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,125评论 1 297
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,028评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,887评论 0 274
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,310评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,533评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,690评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,411评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,004评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,659评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,812评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,693评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,577评论 2 353

推荐阅读更多精彩内容

  • 一、简介 Java在编译时候就必须知道所引用的类所在地方,但是在实际编程中,在某些场合,可能需要引用一个并不在编译...
    千涯秋瑟阅读 560评论 0 3
  • 反射(Reflection)能够让运行于 JVM 中的程序检测和修改运行时的行为。 Class类提供了以下方法 四...
    一只好奇的茂阅读 362评论 0 17
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,651评论 18 139
  • 联合国人类年龄划分新标准,你是在青年还是中年?多少岁,才算青年? 总部设于瑞士日内瓦的联合国世界卫生组织,...
    Stephen潇雨阅读 378评论 0 0
  • 这是洛杉矶东部沙漠中阳光下发生的故事,女神汉娜·罗斯(Hannah Rose)就是沙漠中的玫瑰。 摄影师:Jeff...
    湖南的肖逸飞阅读 1,264评论 0 48