Java反射详解

一、什么是反射?

“反射(Reflection)能够让运行于JVM中的程序检测和修改运行时的行为。反射用于在运行时检测和修改某个对象的结构及其行为。

二、Class反射机制

  • 指的是可以于运行时加载,探知和使用编译期间完全未知的类
  • 程序在运行状态中,,可以动态加载一个只有名称的类, 对于任意一个已经加载的类,都能够知道这个类的所有属性和方法; 对于任意一个对象,都能调用他的任意一个方法和属性
  • 加载完类之后,产生一个Class类型的对象(一个类只有一个Class对象),这个对象包含了完整的类的结构信息,而且这个Class对象就像一面镜子,透过这个镜子看到类的结构,所以被称之为:反射
  • 每个类被加载进入内存之后,系统就会为该类生成一个对应的java.lang.Class对象,通过该Class对象就可以访问到JVM中的这个类

三、为什么需要反射?

  • 在运行时检测对象的类型
  • 动态构造某个类的对象
  • 检测类的属性和方法
  • 任意调用对象的方法
  • 修改构造函数、方法、属性的可见性

四、反射的基本使用

1 Java反射主要API

在java.lang.reflect包中有三个重要的类:

  • Field:描述类的域
  • Method:描述类的方法
  • Constructor:描述类的构造器

对于public域(包括超类成员):

  • getFields
  • getMethods
  • getConstructors

对于其它域(包括私有和受保护的成员,不包括超类成员):

  • gettDeclaredFields
  • gettDeclaredMethods
  • gettDeclaredConstructors

2 获取class对象的三种方式

  • 对象的getClass()方法;
  • 类的.class(最安全/性能最好)属性;
  • 运用Class.forName(String className)动态加载类,className需要是类的全限定名(最常用).

例如对于如下的Person类:

package com.yano.reflect;

public class Person {

    public String name = "default name";
    public int[] array = new int[10];

    public Person() {
        System.out.println(name);
        for (int i = 0; i < array.length; i++) {
            array[i] = i;
        }
    }

    private Person(String name) {
        this.name = name;
        System.out.println(name);
    }

    public void fun() {
        System.out.println("fun");
    }

    public void fun(String name) {
        System.out.println(name);
    }

}

获取Person对象的三种方式为:

// 加载类的3种方法
Class clazz = Class.forName("com.yano.reflect.Person");
Class clazz1 = new Person().getClass();
Class class2 = Person.class;

3 构造函数的获取

获取类的无参构造函数,并实例化类:

Class clazz = Class.forName("com.yano.reflect.Person");
Constructor c = clazz.getConstructor(null);
Person p = (Person) c.newInstance(null);

获取类的含参私有构造函数,并实例化类:

Class clazz = Class.forName("com.yano.reflect.Person");
Constructor c = clazz.getDeclaredConstructor(new Class[] { String.class });
// 由于构造函数是 private 的,所以需要屏蔽Java语言的访问检查
c.setAccessible(true);
Person p = (Person) c.newInstance(new Object[] { "I'm a reflect name!" });

4 类方法的调用

获取并调用类的无参方法:

Class clazz = Class.forName("com.yano.reflect.Person");
Constructor c = clazz.getConstructor(null);
Person p = (Person) c.newInstance(null);
Method method = clazz.getMethod("fun", null);
method.invoke(p, null);

获取并调用类的含参方法:

Class clazz = Class.forName("com.yano.reflect.Person");
Constructor c = clazz.getConstructor(null);
Person p = (Person) c.newInstance(null);
Method method = clazz.getMethod("fun", new Class[] {String.class});
method.invoke(p, new Object[] { "I'm a reflect method!" });

5 类字段的访问

获取类字段:

Class clazz = Class.forName("com.yano.reflect.Person");
Constructor c = clazz.getDeclaredConstructor(new Class[] { String.class });
// 由于构造函数是 private 的,所以需要获取控制权限
c.setAccessible(true);
Person p = (Person) c.newInstance(new Object[] { "I'm a reflect name!" });
Field f = clazz.getField("name");
Object value = f.get(p);
Class type = f.getType();
System.out.println(type);
if (type.equals(String.class)) {
    System.out.println((String) value);
}

参考文章

Java 反射详解
Java反射教程
粗浅看 java反射机制
Java 反射

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 转载注明出处:http://www.jianshu.com/p/2f488de72886 简介 Java在编译时候...
    王三的猫阿德阅读 1,061评论 0 2
  • 今天是二零一六年十一月十七日。这是第十篇简书。 我是喜欢喜剧结尾的,有很长一段时间,在网上找些小说或剧来看来听的时...
    尘___世_美阅读 182评论 0 1
  • 玄奘译本 观自在菩萨,行深般若波罗蜜多时,照见五蕴皆空,度一切苦厄。 舍利子,色不异空,空不异色,色即是空,空即是...
    白语金言阅读 182评论 1 1