java 类反射机制
来源:互联网 发布:舰队collection 知乎 编辑:程序博客网 时间:2024/04/30 13:21
1.什么是反射?
JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法和属性;这种动态获取的信息以及动态调用对象的方法的功能称为Java语言的反射机制。反射(Reflection)是Java程序开发语言的特征之一,它允许运行中的Java程序对自身进行检查,也称自审,并能直接操作程序的内部属性。例如,使用它能获得Java类中各成员的名称并显示出来。
Java的这一能力在实际应用中应用得很多,在其它的程序语言中根本就不存在这一特性。例如,Pascal、C或者C++中就没有办法在程序中获得函数定义相关的信息。
JavaBean是类反射的实际应用之一,它能让一些工具可视化的操作软件组件。这些工具通过类反射动态的载入并取得Java组件(类)的属性。后面学习的各种框架,基本上都会有反射的使用。
2.使用类反射的三个步骤:
例如:用于反射的类,如Method,可以在java.lang.reflect包中找到。使用这些类的时候必须要遵循三个步骤:
第一步:获得你想操作的类的java.lang.Class对象。在运行中的Java程序中,用java.lang.Class类来描述类和接口等。
第二步:调用诸如getDeclaredMethods的方法,取得该类中定义的所有方法的列表。
第三步:使用反射的API来操作这些信息。
<pre name="code" class="html"><pre name="code" class="html"><span style="color:#ff6600;">Classc = Class.forName("java.lang.String");Methodms[]= c.getDeclaredMethods();System.out.println(ms[0].toString());</span>
3.获取Class对象的三种方式
- 通过对象的getClass方法进行获取。这种方式需要具体的类和该类的对象,以及调用getClass方法。
任何数据类型(包括基本数据类型)都具备着一个静态的属性class,通过它可直接获取到该类型对应的Class对象。这种方式要使用具体的类,然后调用类中的静态属性class完成,无需调用方法,性能更好。
通过Class.forName()方法获取。这种方式仅需使用类名,就可以获取该类的Class对象,更有利于扩展。
任何数据类型(包括基本数据类型)都具备着一个静态的属性class,通过它可直接获取到该类型对应的Class对象。这种方式要使用具体的类,然后调用类中的静态属性class完成,无需调用方法,性能更好。
通过Class.forName()方法获取。这种方式仅需使用类名,就可以获取该类的Class对象,更有利于扩展。
</pre></div><pre>
<div><span style="color: rgb(204, 0, 0);"></span><pre name="code" class="java">package cn.hncu.reflect;import org.junit.Test;public class reflectGetClass { public static void main(String[] args) { // getClassObj(); //getClassObj2(); getClassObj3(); } @Test public void test(){ Person p=new Person(); Object obj=(Object)p; Class c=obj.getClass();//运行时多态,new谁调谁 System.out.println(c); System.out.println(int.class);//int System.out.println(Integer.class);//java.lang.Integer System.out.println(Integer.TYPE);//int } private static void getClassObj3() { try { Class c=Class.forName("cn.hncu.reflect.Person");//3333用字符串(类名)---依赖于String类 System.out.println(c); } catch (ClassNotFoundException e) { e.printStackTrace(); } } //所有数据类型(包括基本数据类型)都有一个静态变量class private static void getClassObj2() { Class c=Person.class;//2222通过类名----依赖于Person类 System.out.println(c); } private static void getClassObj() { Person p=new Person();//1111通过对象 Class c=p.getClass(); System.out.println(c); Person p2=new Person();//1111通过对象 Class c2=p.getClass(); System.out.println(c2==c);//同一个类模板,所有对象是共享同一个类模板的 }}
练习(模拟Java内省的功能):
思路:定义一个Model类,里面所有的属性都是private的,然后为每个属性提供getter和setter方法;
再准备一个Map,map的key值都是类里面的属性字段的字符串表示,值任意。
设计一个方法ObjectgetModel(Mapmap,Classcls),传入一个包含所有值的Map,然后再传入Model类的class,那么返回Model类的实例,这个实例里面已经包含好了所有相关的数据。也就是把Map中的数据通过反射,设置回到Model类实例中。
package cn.hncu.reflect.myBeanUtil;import java.lang.reflect.Field;import java.lang.reflect.Method;import java.util.Map;public class MyBeanUtils {public static Object populate(Class cls,Map<String, Object> map) throws ReflectiveOperationException, IllegalAccessException{Object obj=null;//1 用类反射new出对象obj=cls.newInstance();//2 再用类反射对新new的对象设置属性值(必须遵守Java设置规范)--即通过setter方法设置 //2.1遍历出所有该类声明的属性Field flds[]=cls.getDeclaredFields();for(Field fld:flds){//获取该fld对象所代表的属性名String fldName=fld.getName(); //根据属性名,到map中去读取数据,只有数据非空才需要给该属性设置值 Object value=map.get(fld.getName());//返回指定键所映射的值;如果此映射不包含该键的映射关系,则返回 null。if(value==null){System.out.println(fld.getName()+"的数据为空");}else{//如果map中存在对应的属性数据,则由属性名得出它的setter方法的名字String methodName="set"+fldName.substring(0,1).toUpperCase()+fldName.substring(1); //根据方法名和参数的数据类型(其实就是属性的类型),获得Method对象Method method=cls.getDeclaredMethod(methodName, fld.getType()); //调用该method对象所代表的方法Object params[]=new Object[1];params[0]=value;method.invoke(obj, params);}}return obj;}public static<T> T populate2(Class<T> cls,Map<String, Object> map) throws ReflectiveOperationException, IllegalAccessException{T obj=null;//1 用类反射new出对象obj=cls.newInstance();//2 再用类反射对新new的对象设置属性值(必须遵守Java设置规范)--即通过setter方法设置 //2.1遍历出所有该类声明的属性Field flds[]=cls.getDeclaredFields();for(Field fld:flds){//获取该fld对象所代表的属性名String fldName=fld.getName(); //根据属性名,到map中去读取数据,只有数据非空才需要给该属性设置值 Object value=map.get(fld.getName());//返回指定键所映射的值;如果此映射不包含该键的映射关系,则返回 null。if(value==null){System.out.println(fld.getName()+"的数据为空");}else{//如果map中存在对应的属性数据,则由属性名得出它的setter方法的名字String methodName="set"+fldName.substring(0,1).toUpperCase()+fldName.substring(1); //根据方法名和参数的数据类型(其实就是属性的类型),获得Method对象Method method=cls.getDeclaredMethod(methodName, fld.getType()); //调用该method对象所代表的方法Object params[]=new Object[1];params[0]=value;method.invoke(obj, params);}}return obj;}}
- java类反射机制
- java类反射机制
- java类反射机制
- java 类反射机制
- java 类反射机制
- 【反射】JAVA反射机制
- Java中的类反射机制
- Java中的类反射机制
- Java中的类反射机制
- Java中的类反射机制
- Java中的类反射机制
- Java中的类反射机制
- Java中的类反射机制
- Java中的类反射机制
- java中的类反射机制
- Java中的类反射机制
- java类的反射机制
- Java中的类反射机制
- AlertDialog的自定义
- 静态数据竞争检测工具之RELAY-安装和使用
- python 标识符
- 剑指offer——链表中环的入口节点
- 【译】唯快不破:Web 应用的 13 个优化步骤
- java 类反射机制
- 单向链表的创建、节点删除、链表输出、翻转(写法参考计蒜客数据结构课程)
- 提高项目10-编制三角函数表
- 高德地图——查询天气
- ASP.NET解决客户端服务器ID不一致-ClientID与ClientIDMode
- Weex Android SDK源码分析之界面渲染(下)
- 动态数据竞争验证方法(一)
- Electron的第一个应用
- ThreadPoolExecutor运转机制详解