Java 反射机制详解

动态语言

动态语言,是指程序在运行时可以改变其结构:新的函数可以被引进,已有的函数可以被删除等在结构上的变化。比如众所周知的ECMAScript(JavaScript)便是一个动态语言。除此之外如Ruby、Python等也都属于动态语言,而C、C++等语言则不属于动态语言。(引自: 百度百科)

var execString = "alert(Math.floor(Math.random()*10));";

eval(execString);

Class反射机制

指的是可以于运行时加载,探知和使用编译期间完全未知的类.

程序在运行状态中, 可以动态加载一个只有名称的类, 对于任意一个已经加载的类,都能够知道这个类的所有属性和方法; 对于任意一个对象,都能调用他的任意一个方法和属性;

加载完类之后, 在堆内存中会产生一个Class类型的对象(一个类只有一个Class对象), 这个对象包含了完整的类的结构信息,而且这个Class对象就像一面镜子,透过这个镜子看到类的结构,所以被称之为:反射。

Instances of the class Class represent classes and interfaces in a running Java application. An enum is a kind of class and an annotation is a kind of interface. Every array also belongs to a class that is reflected as a Class object that is shared by all arrays with the same element type and number of dimensions(维度). The primitive Java types (boolean, byte, char, short, int, long, float, anddouble), and the keyword void are also represented as Class objects.

每个类被加载进入内存之后,系统就会为该类生成一个对应的java.lang.Class对象,通过该Class对象就可以访问到JVM中的这个类.

Class对象的获取

对象的getClass()方法;

类的.class(最安全/性能最好)属性;

运用Class.forName(String className)动态加载类,className需要是类的全限定名(最常用).

从Class中获取信息

Class类提供了大量的实例方法来获取该Class对象所对应的详细信息,Class类大致包含如下方法,其中每个方法都包含多个重载版本,因此我们只是做简单的介绍,详细请参考JDK文档

获取类内信息

获取内容方法签名

构造器Constructor<T> getConstructor(Class<?>... parameterTypes)

包含的方法Method getMethod(String name, Class<?>... parameterTypes)

包含的属性Field getField(String name)

包含的Annotation<A extends Annotation> A getAnnotation(Class<A> annotationClass)

内部类Class<?>[] getDeclaredClasses()

外部类Class<?> getDeclaringClass()

所实现的接口Class<?>[] getInterfaces()

修饰符int getModifiers()

所在包Package getPackage()

类名String getName()

简称String getSimpleName()

一些判断类本身信息的方法

判断内容方法签名

注解类型?boolean isAnnotation()

使用了该Annotation修饰?boolean isAnnotationPresent(Class<? extends Annotation> annotationClass)

匿名类?boolean isAnonymousClass()

数组?boolean isArray()

枚举?boolean isEnum()

原始类型?boolean isPrimitive()

接口?boolean isInterface()

obj是否是该Class的实例boolean isInstance(Object obj)

使用反射生成并操作对象:

Method Constructor Field这些类都实现了java.lang.reflect.Member接口,程序可以通过Method对象来执行相应的方法,通过Constructor对象来调用对应的构造器创建实例,通过Filed对象直接访问和修改对象的成员变量值.

创建对象

通过反射来生成对象的方式有两种:

使用Class对象的newInstance()方法来创建该Class对象对应类的实例(这种方式要求该Class对象的对应类有默认构造器).

先使用Class对象获取指定的Constructor对象, 再调用Constructor对象的newInstance()方法来创建该Class对象对应类的实例(通过这种方式可以选择指定的构造器来创建实例).

通过第一种方式来创建对象比较常见, 像Spring这种框架都需要根据配置文件(如applicationContext.xml)信息来创建Java对象,从配置文件中读取的只是某个类的全限定名字符串,程序需要根据该字符串来创建对应的实例,就必须使用默认的构造器来反射对象.

下面我们就模拟Spring实现一个简单的对象池, 该对象池会根据文件读取key-value对, 然后创建这些对象, 并放入Map中.

配置文件

{

  "objects": [

    {

      "id": "id1",

      "class": "com.fq.domain.User"

    },

    {

      "id": "id2",

      "class": "com.fq.domain.Bean"

    }

  ]

}

ObjectPool

/**

* Created by w .

*/

public class ObjectPool {

    private Map<String, Object> pool;

    private ObjectPool(Map<String, Object> pool) {

        this.pool = pool;

    }

    private static Object getInstance(String className) throws ClassNotFoundException, IllegalAccessException, InstantiationException {

        return Class.forName(className).newInstance();

    }

    private static JSONArray getObjects(String config) throws IOException {

        Reader reader = new InputStreamReader(ClassLoader.getSystemResourceAsStream(config));

        return JSONObject.parseObject(CharStreams.toString(reader)).getJSONArray("objects");

    }

    // 根据指定的JSON配置文件来初始化对象池

    public static ObjectPool init(String config) {

        try {

            JSONArray objects = getObjects(config);

            ObjectPool pool = new ObjectPool(new HashMap<String, Object>());

            if (objects != null && objects.size() != 0) {

                for (int i = 0; i < objects.size(); ++i) {

                    JSONObject object = objects.getJSONObject(i);

                    if (object == null || object.size() == 0) {

                        continue;

                    }

                    String id = object.getString("id");

                    String className = object.getString("class");

                    pool.putObject(id, getInstance(className));

                }

            }

            return pool;

        } catch (IOException | ClassNotFoundException | InstantiationException | IllegalAccessException e) {

            throw new RuntimeException(e);

        }

    }

    public Object getObject(String id) {

        return pool.get(id);

    }

    public void putObject(String id, Object object) {

        pool.put(id, object);

    }

    public void clear() {

        pool.clear();

    }

}

Client

public class Client {

    @Test

    public void client() {

        ObjectPool pool = ObjectPool.init("config.json");

        User user = (User) pool.getObject("id1");

        System.out.println(user);

        Bean bean = (Bean) pool.getObject("id2");

        System.out.println(bean);

    }

}

User

public class User {

    private int id;

    private String name;

    private String password;

    public int getId() {

        return id;

    }

    public void setId(Integer id) {

        this.id = id;

    }

    public String getName() {

        return name;

    }

    public void setName(String name) {

        this.name = name;

    }

    public String getPassword() {

        return password;

    }

    public void setPassword(String password) {

        this.password = password;

    }

    @Override

    public String toString() {

        return "User{" +

                "id=" + id +

                ", name='" + name + '/'' +

                ", password='" + password + '/'' +

                '}';

    }

}

Bean

public class Bean {

    private Boolean usefull;

    private BigDecimal rate;

    private String name;

    public Boolean getUsefull() {

        return usefull;

    }

    public void setUsefull(Boolean usefull) {

        this.usefull = usefull;

    }

    public BigDecimal getRate() {

        return rate;

    }

    public void setRate(BigDecimal rate) {

        this.rate = rate;

    }

    public String getName() {

        return name;

    }

    public void setName(String name) {

        this.name = name;

    }

    @Override

    public String toString() {

        return "Bean{" +

                "usefull=" + usefull +

                ", rate=" + rate +

                ", name='" + name + '/'' +

                '}';

    }

}

注意: 需要在pom.xml中添加如下依赖:

<dependency>

    <groupId>com.alibaba</groupId>

    <artifactId>fastjson</artifactId>

    <version>1.2.7</version>

</dependency>

<dependency>

    <groupId>com.google.guava</groupId>

    <artifactId>guava</artifactId>

    <version>18.0</version>

</dependency>

调用方法

当获取到某个类对应的Class对象之后, 就可以通过该Class对象的getMethod来获取一个Method数组或Method对象.每个Method对象对应一个方法,在获得Method对象之后,就可以通过调用invoke方法来调用该Method对象对应的方法.

@CallerSensitive

public Object invoke(Object obj, Object... args)

    throws IllegalAccessException, IllegalArgumentException,

      InvocationTargetException

{

    ...

}

下面我们对上面的对象池加强:可以看到Client获取到的对象的成员变量全都是默认值,既然我们已经使用了JSON这么优秀的工具,我们又学习了动态调用对象的方法,那么我们就通过配置文件来给对象设置值(在对象创建时), 新的配置文件形式如下:

{

  "objects": [

    {

      "id": "id1",

      "class": "com.fq.domain.User",

      "fields": [

        {

          "name": "id",

          "value": 101

        },

        {

          "name": "name",

          "value": "feiqing"

        },

        {

          "name": "password",

          "value": "ICy5YqxZB1uWSwcVLSNLcA=="

        }

      ]

    },

    {

      "id": "id2",

      "class": "com.fq.domain.Bean",

      "fields": [

        {

          "name": "usefull",

          "value": true

        },

        {

          "name": "rate",

          "value": 3.14

        },

        {

          "name": "name",

          "value": "bean-name"

        }

      ]

    },

    {

      "id": "id3",

      "class": "com.fq.domain.ComplexBean",

      "fields": [

        {

          "name": "name",

          "value": "complex-bean-name"

        },

        {

          "name": "refBean",

          "ref": "id2"

        }

      ]

    }

  ]

}

其中fields代表该Bean所包含的属性,name为属性名称,value为属性值(属性类型为JSON支持的类型),ref代表引用一个对象(也就是属性类型为Object,但是一定要引用一个已经存在了的对象)

/**

* @author it`w

*

*/

public class ObjectPool {

    private Map<String, Object> pool;

    private ObjectPool(Map<String, Object> pool) {

        this.pool = pool;

    }

    private static JSONArray getObjects(String config) throws IOException {

        Reader reader = new InputStreamReader(ClassLoader.getSystemResourceAsStream(config));

        return JSONObject.parseObject(CharStreams.toString(reader)).getJSONArray("objects");

    }

    private static Object getInstance(String className, JSONArray fields)

            throws ClassNotFoundException, NoSuchMethodException,

            IllegalAccessException, InstantiationException, InvocationTargetException {

        // 配置的Class

        Class<?> clazz = Class.forName(className);

        // 目标Class的实例对象

        Object targetObject = clazz.newInstance();

        if (fields != null && fields.size() != 0) {

            for (int i = 0; i < fields.size(); ++i) {

                JSONObject field = fields.getJSONObject(i);

                // 需要设置的成员变量名

                String fieldName = field.getString("name");

                // 需要设置的成员变量的值

                Object fieldValue;

                if (field.containsKey("value")) {

                    fieldValue = field.get("value");

                } else if (field.containsKey("ref")) {

                    String refBeanId = field.getString("ref");

                    fieldValue = OBJECTPOOL.getObject(refBeanId);

                } else {

                    throw new RuntimeException("neither value nor ref");

                }

                String setterName = "set" +

                        fieldName.substring(0, 1).toUpperCase() +

                        fieldName.substring(1);

                // 需要设置的成员变量的setter方法

                Method setterMethod = clazz.getMethod(setterName, fieldValue.getClass());

                // 调用setter方法将值设置进去

                setterMethod.invoke(targetObject, fieldValue);

            }

        }

        return targetObject;

    }

    private static ObjectPool OBJECTPOOL;

    // 创建一个对象池的实例(保证是多线程安全的)

    private static void initSingletonPool() {

        if (OBJECTPOOL == null) {

            synchronized (ObjectPool.class) {

                if (OBJECTPOOL == null) {

                    OBJECTPOOL = new ObjectPool(new ConcurrentHashMap<String, Object>());

                }

            }

        }

    }

    // 根据指定的JSON配置文件来初始化对象池

    public static ObjectPool init(String config) {

        // 初始化pool

        initSingletonPool();

        try {

            JSONArray objects = getObjects(config);

            for (int i = 0; objects != null && i < objects.size(); ++i) {

                JSONObject object = objects.getJSONObject(i);

                if (object == null || object.size() == 0) {

                    continue;

                }

                String id = object.getString("id");

                String className = object.getString("class");

                // 初始化bean并放入池中

                OBJECTPOOL.putObject(id, getInstance(className, object.getJSONArray("fields")));

            }

            return OBJECTPOOL;

        } catch (IOException | ClassNotFoundException |

                InstantiationException | IllegalAccessException |

                NoSuchMethodException | InvocationTargetException e) {

            throw new RuntimeException(e);

        }

    }

    public Object getObject(String id) {

        return pool.get(id);

    }

    public void putObject(String id, Object object) {

        pool.put(id, object);

    }

    public void clear() {

        pool.clear();

    }

}

Client

public class Client {

    @Test

    public void client() {

        ObjectPool pool = ObjectPool.init("config.json");

        User user = (User) pool.getObject("id1");

        System.out.println(user);

        Bean bean = (Bean) pool.getObject("id2");

        System.out.println(bean);

        ComplexBean complexBean = (ComplexBean) pool.getObject("id3");

        System.out.println(complexBean);

    }

}

ComplexBean

public class ComplexBean {

    private String name;

    private Bean refBean;

    public String getName() {

        return name;

    }

    public void setName(String name) {

        this.name = name;

    }

    public Bean getRefBean() {

        return refBean;

    }

    public void setRefBean(Bean refBean) {

        this.refBean = refBean;

    }

    @Override

    public String toString() {

        return "ComplexBean{" +

                "name='" + name + '/'' +

                ", refBean=" + refBean +

                '}';

    }

}

Spring框架就是通过这种方式将成员变量值以及依赖对象等都放在配置文件中进行管理的,从而实现了较好地解耦(不过Spring是通过XML作为配置文件).

访问成员变量

通过Class对象的的getField()方法可以获取该类所包含的全部或指定的成员变量Field,Filed提供了如下两组方法来读取和设置成员变量值.

getXxx(Object obj): 获取obj对象的该成员变量的值, 此处的Xxx对应8中基本类型,如果该成员变量的类型是引用类型, 则取消get后面的Xxx;

setXxx(Object obj, Xxx val): 将obj对象的该成员变量值设置成val值.此处的Xxx对应8种基本类型, 如果该成员类型是引用类型, 则取消set后面的Xxx;

注: getDeclaredXxx方法可以获取所有的成员变量,无论private/public;

/**

* @author w

*/

public class Client {

    @Test

    public void client() throws NoSuchFieldException, IllegalAccessException {

        User user = new User();

        Field idFiled = User.class.getDeclaredField("id");

        setAccessible(idFiled);

        idFiled.setInt(user, 46);

        Field nameFiled = User.class.getDeclaredField("name");

        setAccessible(nameFiled);

        nameFiled.set(user, "feiqing");

        Field passwordField = User.class.getDeclaredField("password");

        setAccessible(passwordField);

        passwordField.set(user, "ICy5YqxZB1uWSwcVLSNLcA==");

        System.out.println(user);

    }

    private void setAccessible(AccessibleObject object) {

        object.setAccessible(true);

    }

}

使用反射获取泛型信息

为了通过反射操作泛型以迎合实际开发的需要, Java新增了java.lang.reflect.ParameterizedTypejava.lang.reflect.GenericArrayTypejava.lang.reflect.TypeVariablejava.lang.reflect.WildcardType几种类型来代表不能归一到Class类型但是又和原始类型同样重要的类型.

类型含义

ParameterizedType一种参数化类型, 比如Collection<String>

GenericArrayType一种元素类型是参数化类型或者类型变量的数组类型

TypeVariable各种类型变量的公共接口

WildcardType一种通配符类型表达式, 如?? extends Number? super Integer

其中, 我们可以使用ParameterizedType来获取泛型信息.

public class Client {

    private Map<String, Object> objectMap;

    public void test(Map<String, User> map, String string) {

    }

    public Map<User, Bean> test() {

        return null;

    }

    /**

    * 测试属性类型

    *

    * @throws NoSuchFieldException

    */

    @Test

    public void testFieldType() throws NoSuchFieldException {

        Field field = Client.class.getDeclaredField("objectMap");

        Type gType = field.getGenericType();

        // 打印type与generic type的区别

        System.out.println(field.getType());

        System.out.println(gType);

        System.out.println("**************");

        if (gType instanceof ParameterizedType) {

            ParameterizedType pType = (ParameterizedType) gType;

            Type[] types = pType.getActualTypeArguments();

            for (Type type : types) {

                System.out.println(type.toString());

            }

        }

    }

    /**

    * 测试参数类型

    *

    * @throws NoSuchMethodException

    */

    @Test

    public void testParamType() throws NoSuchMethodException {

        Method testMethod = Client.class.getMethod("test", Map.class, String.class);

        Type[] parameterTypes = testMethod.getGenericParameterTypes();

        for (Type type : parameterTypes) {

            System.out.println("type -> " + type);

            if (type instanceof ParameterizedType) {

                Type[] actualTypes = ((ParameterizedType) type).getActualTypeArguments();

                for (Type actualType : actualTypes) {

                    System.out.println("/tactual type -> " + actualType);

                }

            }

        }

    }

    /**

    * 测试返回值类型

    *

    * @throws NoSuchMethodException

    */

    @Test

    public void testReturnType() throws NoSuchMethodException {

        Method testMethod = Client.class.getMethod("test");

        Type returnType = testMethod.getGenericReturnType();

        System.out.println("return type -> " + returnType);

        if (returnType instanceof ParameterizedType) {

            Type[] actualTypes = ((ParameterizedType) returnType).getActualTypeArguments();

            for (Type actualType : actualTypes) {

                System.out.println("/tactual type -> " + actualType);

            }

        }

    }

}

使用反射获取注解

使用反射获取注解信息的相关介绍, 请参看我的博客Java注解实践

反射性能测试

Method/Constructor/Field/Element都继承了AccessibleObject,AccessibleObject类中有一个setAccessible方法:

public void setAccessible(boolean flag) throws SecurityException {

    ...

}

该方法有两个作用:

1. 启用/禁用访问安全检查开关:值为true,则指示反射的对象在使用时取消Java语言访问检查;值为false,则指示应该实施Java语言的访问检查;

2. 可以禁止安全检查, 提高反射的运行效率.

/**

* @author w

*/

public class TestReflect {

    @Before

    public void testNoneReflect() {

        User user = new User();

        long start = System.currentTimeMillis();

        for (long i = 0; i < Integer.MAX_VALUE; ++i) {

            user.getName();

        }

        long count = System.currentTimeMillis() - start;

        System.out.println("没有反射, 共消耗 <" + count + "> 毫秒");

    }

    @Test

    public void testNotAccess() throws ClassNotFoundException, IllegalAccessException, InstantiationException, NoSuchMethodException, InvocationTargetException {

        User user = new User();

        Method method = Class.forName("com.fq.domain.User").getMethod("getName");

        long start = System.currentTimeMillis();

        for (long i = 0; i < Integer.MAX_VALUE; ++i) {

            method.invoke(user, null);

        }

        long count = System.currentTimeMillis() - start;

        System.out.println("没有访问权限, 共消耗 <" + count + "> 毫秒");

    }

    @After

    public void testUseAccess() throws ClassNotFoundException, IllegalAccessException, InstantiationException, NoSuchMethodException, InvocationTargetException {

        User user = new User();

        Method method = Class.forName("com.fq.domain.User").getMethod("getName");

        method.setAccessible(true);

        long start = System.currentTimeMillis();

        for (long i = 0; i < Integer.MAX_VALUE; ++i) {

            method.invoke(user, null);

        }

        long count = System.currentTimeMillis() - start;

        System.out.println("有访问权限, 共消耗 <" + count + "> 毫秒");

    }

}

执行上面程序,在我的机器上会有如下结果:

机器配置信息如下:

可以看到使用反射会比直接调用慢3000毫秒,但是前提是该方法会执行20E+次(而且服务器的性能也肯定比我的机器要高),因此在我们的实际开发中,其实是不用担心反射机制带来的性能消耗的,而且禁用访问权限检查,也会有性能的提升。

欢迎关注微信公众号:Java的学习之路

里面资料非常全,从java初级到高级都有,视频,电子书,面试宝典,简历模板,经典案例,源码分析程序员故事以及解决bug方法。。。。应有尽有,可以推荐大家一起学习下!!

​​​​

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

推荐阅读更多精彩内容