反射机制 备选

一、java的反射机制浅谈

最近研究java研究得很给力,主要以看博文为学习方式。以下是我对java的反射机制所产生的一些感悟,希望各位童鞋看到失误之处不吝指出。受到各位指教之处,如若让小生好好感动,说不定会请各位吃饭哦!

1.何谓反射机制

根据网文,java中的反射机制可以如此定义:

JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法;这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制。

2.反射机制如何实现

谈到反射机制,最诱人的莫过于“动态”二字了。接触过C语言的童鞋们都知道,C语言中也有个和“动态”搭上边的函数:malloc()函数。其实这里的两个动态是一个意思,都指的是非编译时处理,抑或运行时处理。这种机制,可以让程序的弹性增加不少,因为借由此机制,客户可以在程序运行时改变一些他关心的性质:分配内存(当然他可能 完全不知道这么做了),调用某个类(当然他还是被蒙在鼓里)等。

下面我们就聊聊java中动态机制是如何实现的。

上一篇文章中提到了java的类的加载问题,但没有更深入地解释其运行机制,在这里就先谈谈这个问题。

首先不得不提到的是java.lang.Class这个类。

有这么一段话:

Java程序在运行时,Java运行时系统一直对所有的对象进行所谓的运行时类型标识。这项信息纪录了每个对象所属的类。虚拟机通常使用运行时类型信息选准正确方法去执行,用来保存这些类型信息的类是Class类。

也就是说,ClassLoader找到了需要调用的类时(java为了调控内存的调用消耗,类的加载都在需要时再进行,很抠但是很有效),就会加载它,然后根据.class文件内记载的类信息来产生一个与该类相联系的独一无二的Class对象。该Class对象记载了该类的字段,方法等等信息。以后jvm要产生该类的实例,就是根据内存中存在的该Class类所记载的信息(Class对象应该和我所了解的其他类一样会在堆内存内产生、消亡)来进行。

而java中的Class类对象是可以人工自然性的(也就是说开放的)得到的(虽然你无法像其他类一样运用构造器来得到它的实例,因为

Class对象都是jvm产生的。不过话说回来,客户产生的话也是无意义的),而且,更伟大的是,基于这个基础,java实现了反射机制。

获取Class对象有三种方式:

1.通过Object类的getClass()方法。例如:

Class c1 = new String("").getClass();

2.通过Class类的静态方法——forName()来实现:

Class c2 = Class.forName("MyObject");

3.如果T是一个已定义的类型的话,在java中,它的.class文件名:T.class就代表了与其匹配的Class对象,例如:

Class c3 = Manager.class;

Class c4 = int.class;

Class c5 = Double[].class;

这里需要解释一下3:请记住一句话,java中,一切皆对象。也就是说,基本类型int float 等也会在jvm的内存池像其他类型一样中生成

一个Class对象。而数组等组合型数据类型也是会生成一个Class对象的,而且更令人惊讶的是,java中数组的本来面目其实就是某个类,惊讶

中的惊讶是,含有相同元素的相同维数的数组还会共同享用同一个Class对象!其实根据我的臆想,数组的length性质应该就保存在这个Class

对象里面。

Class类中存在以下几个重要的方法:

1.getName()

一个Class对象描述了一个特定类的特定属性,而这个方法就是返回String形式的该类的简要描述。由于历史原因,对数组的Class对象

调用该方法会产生奇怪的结果。

2.newInstance()

该方法可以根据某个Class对象产生其对应类的实例。需要强调的是,它调用的是此类的默认构造方法。例如:

MyObject x = new MyObject();

MyObject y = x.getClass().newInstance();

3.getClassLoader()

返回该Class对象对应的类的类加载器。

4.getComponentType()

该方法针对数组对象的Class对象,可以得到该数组的组成元素所对应对象的Class对象。例如:

int[] ints = new int[]{1,2,3};

Class class1 = ints.getClass();

Class class2 = class1.getComponentType();

而这里得到的class2对象所对应的就应该是int这个基本类型的Class对象。

5.getSuperClass()

返回某子类所对应的直接父类所对应的Class对象。

6.isArray()

判定此Class对象所对应的是否是一个数组对象。

好啦,现在对Class这个类应该有了一个大致的了解,下面就给出一个反射机制的典型例子供各位分析:

[java] view plaincopyprint?

  1. import java.lang.reflect.Array;
  2. import java.lang.reflect.Constructor;
  3. import java.lang.reflect.Field;
  4. import java.lang.reflect.Method;
  5. /**
  6. * Java Reflection Cookbook
  7. *
  8. * @author Michael Lee
  9. * @since 2006-8-23
  10. * @version 0.1a
  11. */
  12. public class Reflection {
  13. /**
  14. * 得到某个对象的公共属性
  15. *
  16. * @param owner, fieldName
  17. * @return 该属性对象
  18. * @throws Exception
  19. *
  20. */
  21. public Object getProperty(Object owner, String fieldName) throws Exception {
  22. Class ownerClass = owner.getClass();
  23. Field field = ownerClass.getField(fieldName);
  24. Object property = field.get(owner);
  25. return property;
  26. }
  27. /**
  28. * 得到某类的静态公共属性
  29. *
  30. * @param className 类名
  31. * @param fieldName 属性名
  32. * @return 该属性对象
  33. * @throws Exception
  34. */
  35. public Object getStaticProperty(String className, String fieldName)
  36. throws Exception {
  37. Class ownerClass = Class.forName(className);
  38. Field field = ownerClass.getField(fieldName);
  39. Object property = field.get(ownerClass);
  40. return property;
  41. }
  42. /**
  43. * 执行某对象方法
  44. *
  45. * @param owner
  46. * 对象
  47. * @param methodName
  48. * 方法名
  49. * @param args
  50. * 参数
  51. * @return 方法返回值
  52. * @throws Exception
  53. */
  54. public Object invokeMethod(Object owner, String methodName, Object[] args)
  55. throws Exception {
  56. Class ownerClass = owner.getClass();
  57. Class[] argsClass = new Class[args.length];
  58. for (int i = 0, j = args.length; i < j; i++) {
  59. argsClass[i] = args[i].getClass();
  60. }
  61. Method method = ownerClass.getMethod(methodName, argsClass);
  62. return method.invoke(owner, args);
  63. }
  64. /**
  65. * 执行某类的静态方法
  66. *
  67. * @param className
  68. * 类名
  69. * @param methodName
  70. * 方法名
  71. * @param args
  72. * 参数数组
  73. * @return 执行方法返回的结果
  74. * @throws Exception
  75. */
  76. public Object invokeStaticMethod(String className, String methodName,
  77. Object[] args) throws Exception {
  78. Class ownerClass = Class.forName(className);
  79. Class[] argsClass = new Class[args.length];
  80. for (int i = 0, j = args.length; i < j; i++) {
  81. argsClass[i] = args[i].getClass();
  82. }
  83. Method method = ownerClass.getMethod(methodName, argsClass);
  84. return method.invoke(null, args);
  85. }
  86. /**
  87. * 新建实例
  88. *
  89. * @param className
  90. * 类名
  91. * @param args
  92. * 构造函数的参数
  93. * @return 新建的实例
  94. * @throws Exception
  95. */
  96. public Object newInstance(String className, Object[] args) throws Exception {
  97. Class newoneClass = Class.forName(className);
  98. Class[] argsClass = new Class[args.length];
  99. for (int i = 0, j = args.length; i < j; i++) {
  100. argsClass[i] = args[i].getClass();
  101. }
  102. Constructor cons = newoneClass.getConstructor(argsClass);
  103. return cons.newInstance(args);
  104. }
  105. /**
  106. * 是不是某个类的实例
  107. * @param obj 实例
  108. * @param cls 类
  109. * @return 如果 obj 是此类的实例,则返回 true
  110. */
  111. public boolean isInstance(Object obj, Class cls) {
  112. return cls.isInstance(obj);
  113. }
  114. /**
  115. * 得到数组中的某个元素
  116. * @param array 数组
  117. * @param index 索引
  118. * @return 返回指定数组对象中索引组件的值
  119. */
  120. public Object getByArray(Object array, int index) {
  121. return Array.get(array,index);
  122. }
  123. }

一、java的反射机制浅谈

最近研究java研究得很给力,主要以看博文为学习方式。以下是我对java的反射机制所产生的一些感悟,希望各位童鞋看到失误之处不吝指出。受到各位指教之处,如若让小生好好感动,说不定会请各位吃饭哦!

1.何谓反射机制

根据网文,java中的反射机制可以如此定义:

JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法;这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制。

2.反射机制如何实现

谈到反射机制,最诱人的莫过于“动态”二字了。接触过C语言的童鞋们都知道,C语言中也有个和“动态”搭上边的函数:malloc()函数。其实这里的两个动态是一个意思,都指的是非编译时处理,抑或运行时处理。这种机制,可以让程序的弹性增加不少,因为借由此机制,客户可以在程序运行时改变一些他关心的性质:分配内存(当然他可能 完全不知道这么做了),调用某个类(当然他还是被蒙在鼓里)等。

下面我们就聊聊java中动态机制是如何实现的。

上一篇文章中提到了java的类的加载问题,但没有更深入地解释其运行机制,在这里就先谈谈这个问题。

首先不得不提到的是java.lang.Class这个类。

有这么一段话:

Java程序在运行时,Java运行时系统一直对所有的对象进行所谓的运行时类型标识。这项信息纪录了每个对象所属的类。虚拟机通常使用运行时类型信息选准正确方法去执行,用来保存这些类型信息的类是Class类。

也就是说,ClassLoader找到了需要调用的类时(java为了调控内存的调用消耗,类的加载都在需要时再进行,很抠但是很有效),就会加载它,然后根据.class文件内记载的类信息来产生一个与该类相联系的独一无二的Class对象。该Class对象记载了该类的字段,方法等等信息。以后jvm要产生该类的实例,就是根据内存中存在的该Class类所记载的信息(Class对象应该和我所了解的其他类一样会在堆内存内产生、消亡)来进行。

而java中的Class类对象是可以人工自然性的(也就是说开放的)得到的(虽然你无法像其他类一样运用构造器来得到它的实例,因为

Class对象都是jvm产生的。不过话说回来,客户产生的话也是无意义的),而且,更伟大的是,基于这个基础,java实现了反射机制。

获取Class对象有三种方式:

1.通过Object类的getClass()方法。例如:

Class c1 = new String("").getClass();

2.通过Class类的静态方法——forName()来实现:

Class c2 = Class.forName("MyObject");

3.如果T是一个已定义的类型的话,在java中,它的.class文件名:T.class就代表了与其匹配的Class对象,例如:

Class c3 = Manager.class;

Class c4 = int.class;

Class c5 = Double[].class;

这里需要解释一下3:请记住一句话,java中,一切皆对象。也就是说,基本类型int float 等也会在jvm的内存池像其他类型一样中生成

一个Class对象。而数组等组合型数据类型也是会生成一个Class对象的,而且更令人惊讶的是,java中数组的本来面目其实就是某个类,惊讶

中的惊讶是,含有相同元素的相同维数的数组还会共同享用同一个Class对象!其实根据我的臆想,数组的length性质应该就保存在这个Class

对象里面。

Class类中存在以下几个重要的方法:

1.getName()

一个Class对象描述了一个特定类的特定属性,而这个方法就是返回String形式的该类的简要描述。由于历史原因,对数组的Class对象

调用该方法会产生奇怪的结果。

2.newInstance()

该方法可以根据某个Class对象产生其对应类的实例。需要强调的是,它调用的是此类的默认构造方法。例如:

MyObject x = new MyObject();

MyObject y = x.getClass().newInstance();

3.getClassLoader()

返回该Class对象对应的类的类加载器。

4.getComponentType()

该方法针对数组对象的Class对象,可以得到该数组的组成元素所对应对象的Class对象。例如:

int[] ints = new int[]{1,2,3};

Class class1 = ints.getClass();

Class class2 = class1.getComponentType();

而这里得到的class2对象所对应的就应该是int这个基本类型的Class对象。

5.getSuperClass()

返回某子类所对应的直接父类所对应的Class对象。

6.isArray()

判定此Class对象所对应的是否是一个数组对象。

好啦,现在对Class这个类应该有了一个大致的了解,下面就给出一个反射机制的典型例子供各位分析:

[java] view plaincopyprint?

  1. import java.lang.reflect.Array;
  2. import java.lang.reflect.Constructor;
  3. import java.lang.reflect.Field;
  4. import java.lang.reflect.Method;
  5. /**
  6. * Java Reflection Cookbook
  7. *
  8. * @author Michael Lee
  9. * @since 2006-8-23
  10. * @version 0.1a
  11. */
  12. public class Reflection {
  13. /**
  14. * 得到某个对象的公共属性
  15. *
  16. * @param owner, fieldName
  17. * @return 该属性对象
  18. * @throws Exception
  19. *
  20. */
  21. public Object getProperty(Object owner, String fieldName) throws Exception {
  22. Class ownerClass = owner.getClass();
  23. Field field = ownerClass.getField(fieldName);
  24. Object property = field.get(owner);
  25. return property;
  26. }
  27. /**
  28. * 得到某类的静态公共属性
  29. *
  30. * @param className 类名
  31. * @param fieldName 属性名
  32. * @return 该属性对象
  33. * @throws Exception
  34. */
  35. public Object getStaticProperty(String className, String fieldName)
  36. throws Exception {
  37. Class ownerClass = Class.forName(className);
  38. Field field = ownerClass.getField(fieldName);
  39. Object property = field.get(ownerClass);
  40. return property;
  41. }
  42. /**
  43. * 执行某对象方法
  44. *
  45. * @param owner
  46. * 对象
  47. * @param methodName
  48. * 方法名
  49. * @param args
  50. * 参数
  51. * @return 方法返回值
  52. * @throws Exception
  53. */
  54. public Object invokeMethod(Object owner, String methodName, Object[] args)
  55. throws Exception {
  56. Class ownerClass = owner.getClass();
  57. Class[] argsClass = new Class[args.length];
  58. for (int i = 0, j = args.length; i < j; i++) {
  59. argsClass[i] = args[i].getClass();
  60. }
  61. Method method = ownerClass.getMethod(methodName, argsClass);
  62. return method.invoke(owner, args);
  63. }
  64. /**
  65. * 执行某类的静态方法
  66. *
  67. * @param className
  68. * 类名
  69. * @param methodName
  70. * 方法名
  71. * @param args
  72. * 参数数组
  73. * @return 执行方法返回的结果
  74. * @throws Exception
  75. */
  76. public Object invokeStaticMethod(String className, String methodName,
  77. Object[] args) throws Exception {
  78. Class ownerClass = Class.forName(className);
  79. Class[] argsClass = new Class[args.length];
  80. for (int i = 0, j = args.length; i < j; i++) {
  81. argsClass[i] = args[i].getClass();
  82. }
  83. Method method = ownerClass.getMethod(methodName, argsClass);
  84. return method.invoke(null, args);
  85. }
  86. /**
  87. * 新建实例
  88. *
  89. * @param className
  90. * 类名
  91. * @param args
  92. * 构造函数的参数
  93. * @return 新建的实例
  94. * @throws Exception
  95. */
  96. public Object newInstance(String className, Object[] args) throws Exception {
  97. Class newoneClass = Class.forName(className);
  98. Class[] argsClass = new Class[args.length];
  99. for (int i = 0, j = args.length; i < j; i++) {
  100. argsClass[i] = args[i].getClass();
  101. }
  102. Constructor cons = newoneClass.getConstructor(argsClass);
  103. return cons.newInstance(args);
  104. }
  105. /**
  106. * 是不是某个类的实例
  107. * @param obj 实例
  108. * @param cls 类
  109. * @return 如果 obj 是此类的实例,则返回 true
  110. */
  111. public boolean isInstance(Object obj, Class cls) {
  112. return cls.isInstance(obj);
  113. }
  114. /**
  115. * 得到数组中的某个元素
  116. * @param array 数组
  117. * @param index 索引
  118. * @return 返回指定数组对象中索引组件的值
  119. */
  120. public Object getByArray(Object array, int index) {
  121. return Array.get(array,index);
  122. }
  123. }
时间: 2024-11-08 21:09:40

反射机制 备选的相关文章

Android小例子:使用反射机制来读取图片制作一个图片浏览器

效果图: 工程文件夹: 该例子可供于新手参考练习,如果有哪里不对的地方,望指正>-< <黑幕下的人> java代码(MainActivity.java): package com.example.imageswitchtest; import java.lang.reflect.Field; import android.app.Activity; import android.os.Bundle; import android.util.Log; import android.v

java反射机制(一)—— 利用反射机制实例化对象

一.Java有着一个非常突出的动态相关机制:Reflection,用在Java身上指的是我们可以于运行时加载.探知.使用编译期间完全未知的classes.换句话说,Java程序可以加载一个运行时才得知名称的class,获悉其完整构造(但不包括methods定义),并生成其对象实体.或对其fields设值.或唤起其methods.(度娘文库是这么说的) 二.这篇文章主要介绍一下通过反射机制去实例化一个类的对象,然后调用其方法.本文主要介绍两种方式,第一种就是通过构造函数来实例化,第二种就是通过Cl

Java反射机制

Java的反射机制概念 主要是指程序可以访问,检测和修改它本身状态或行为的一种能力,并能根据自身行为的状态和结果,调整或修改应用所描述行为的状态和相关的语义. 反射是java中一种强大的工具,能够使我们很方便的创建灵活的代码,这些代码可以再运行时装配,无需在组件之间进行源代码链接.但是反射使用不当会成本很高! 反射的作用 1 可以反编译将class文件编译成java文件 2 可以通过反射机制访问Java对象的属性,方法,构造方法等 反射机制使用步骤 1 得到要调用类的class 2 通过得到的c

【java】java反射机制,动态获取对象的属性和对应的参数值,并属性按照字典序排序,Field.setAccessible()方法的说明【可用于微信支付 签名生成】

方法1:通过get()方法获取属性值 package com.sxd.test.controller; public class FirstCa{ private Integer num; private String name; private Boolean flag; public Integer getNum() { return num; } public void setNum(Integer num) { this.num = num; } public String getNam

使用反射机制调用属性和私有成员与代理模式的介绍

使用反射机制调用属性: 通过反射机制可以获得类的属性,获得到的属性同样的可以进行赋值.得值操作,调用getField方法并传递属性的名称可以获得[学Java,到凯哥学堂kaige123.com]指定的属性,调用getFields方法则可以获得全部属性,但是这种方式不能获得私有属性: 代码示例: Student类示例: 运行结果: 从运行结果可以看出只拿出了公开的属性,私有的属性拿不到. 使用反射机制调用私有成员: 1.调用私有属性 在反射机制里调用私有属性需要通过getDeclaredField

反射机制

反射机制: 反射机制是能够帮助我们把代码变得更加灵活,可扩展性更高,俗称"软编程.软写法".例如:有一个文件里面有一些值,想要把这些值赋值到一个Student类的属性中,按照以前所学到的知识点,只能是通过文件流将文件里的值读取出来,然后再通过set方法将这些值都赋给Student类的属性.但是,这种方式是将代码写死了,在这之后会有一个问题就是如果文件里的内容发生变动,或者要把这个Student类更换成Dog类的话,就要重新修改代码.程序写好了部署到服务器上运行了,总不能时不时就打开来修

Python学习心得(六) 反射机制、装饰器

1.反射机制 #/usr/bin/env python # -*- coding:utf-8 -*- ''' Python反射机制的核心本质:利用字符串的形式去对象(模块)中操作(查找/获取/删除/添加)成员,一种基于字符串的事件驱动 通俗讲就是通过用户传入url的不同,调用不同的模块函数,好多比较流行的web框架都是通过反射的机制,根据url的不同指向不同的模块 getattr(),hasattr(),setattr(),delattr()对模块的修改都在内存中进行,并不会影响文件中的真实内容

Java 反射机制

使用 Java 反射机制可以在运行时期检查 Java 类的信息,检查 Java 类的信息往往是你在使用 Java 反射机制的时候所做的第一件事情,通过获取类的信息你可以获取以下相关的内容: Class 对象 类名 修饰符 包信息 父类 实现的接口 构造器 方法 变量 注解 除了上述这些内容,还有很多的信息你可以通过反射机制获得,如果你想要知道全部的信息你可以查看相应的文档 JavaDoc for java.lang.Class 里面有详尽的描述. 在本节中我们会简短的涉及上述所提及的信息,上述的

Java的反射机制简述

反射机制是Java语言中一个非常重要的特性,它允许程序在运行时进行自我检查,同时也允许内部的成员进行操作.虽然这个特性在实际开发中使用的不多,但是像Pascal.C和C++等语言根本没有提供这样的特性.由于反射机制能够实现在运行时对类进行装载,因此能够增加程序的灵活性,但是不恰当地使用反射机制也会严重影响系统的性能. 具体而言,反射机制提供的功能主要有: 1.得到一个对象所属的类: 2.获取一个类的所有成员变量和方法: 3.在运行时创建对象: 4.在运行时调用对象的方法. 其实,反射机制非常重要