开发文章

Reflect Java反射机制

最近发现好多框架底层的实现与Java的reflect和cglib有关,看过原理后找了几篇经典的文档,以供后来复习使用


 前言:我 们知道,类和类的成员变量及方法都是要求有权限控制的(public、protected、private);而当类中的信息封装为私有时,外部对该类中 私有的信息是没有访问权限的,也就是说当该类里的内容信息均受private权限控制时,外部想要获取和处理该类里的私有信息是几乎不可能的;但是,有时 候这种需求是有的,而当我们非得需要去动用别的类里封装的私有信息时,java的反射机制就起到了非常关键的作用了;

java反射机制的实现主要由三个类来主导:它们分别是Class、Field、Method;

 1. Class:

     java 在编译和运行时,会将需要被编译和运行的所有类加载进类加载器,每个类被加载之后,系统就会为该类生成一个对应的Class对象,通过该Class对象就 可以访问到java虚拟机中的这个类,进而在运行时对这个被访问的类进行信息的获取和处理(当然,不管被访问的这个类里的信息是否私有的);通俗的 讲,Class对象间接代表了它对应的类,通过这个Class对象,我们就可以去执行反射机制的实现;

       获取Class对象主要有三种方式:

       1). 调用Class类的forName(String name)静态方法,参数name为Class对应的类的全名(包括包名);

           比如我们要创建Gesture这个类对应的Class对象:

                  Class<Gesture> mClass = Class.forName("Android.gesture.Gesture");

           android.gesture为Gesture的包名,Class<Gesture>中的Gesture表示得到的是Gesture类型 对应的Class对象,<>中的Gesture也可为通配符?表示,如:Class<?>mClass = Class.forName("android.gesture.Gesture");

       2). 调用类的class属性得到类对应的Class对象。如:Class<?>mClass = Gesture.class; (一般建议用这种方式得到Class对象)

       3).调用类的实例化对象的getClass()方法。 特别说明的是,getClass()是java类的始祖Object类的方法,所以,所有java对象都可以调用该方法;如mGesture是 Gesture类型的对象,Class<?> mClass = mGesture.getClass()得到的是Gesture类对应的Class对象

    那么在得到对应类的Class对象对应后,我们就可以通过该Class对象得到它所对应的类的一些信息,比如该类的构造函数、成员(属性)、方法(函数);

       Class类提供的相关接口介绍:(详细参看API)

接口 返回类型 接口功能实现
getPackage()
Package 得到目标类的包名对应的Package对象
getCanonicalName()
String 得到目标类的全名(包名+类名)
getName() String 同getCanonicalName()
getClassLoader()
ClassLoader
得到加载目标类的ClassLoader对象
getClasses()
Class<?>[] 得到目标类中的所有的public内部类以及public内部接口所对应的Class对象
getDeclaredClasses()
Class<?>[]
同getClasses(),但不局限于public修饰,只要是目标类中声明的内部类和接口均可
getConstructors()
Constructor<?>[]
得到目标类的所有public构造函数对应的Constructor对象
getDeclaredConstructors()
Constructor<?>[]
同getConstructors(),但不局限于public修饰,只要是目标类中声明的构造函数均可
getField(String arg)
Field 得到目标类中指定的某个public属性对应的Field对象
getDeclaredField(String arg)
Field 同getField,但不局限于public修饰,只要是目标类中声明的属性均可
getFields()
Field[] 得到目标类中所有的public属性对应的Field对象
getDeclaredFields()
Field[] 同getFields(),但不局限于public修饰的属性
getMethod(String arg0, Class<?>... arg1)
method 得到目标类中指定的某个public方法对应的Method对象
getDeclaredMethod(String arg0, Class<?>... arg1)
Method 同getMethod,但不局限于public修饰的方法
getMethods()
Method[] 得到目标类中所有的public方法对应的Method对象
getDeclaredMethods()
Method[] 同getMethods(),但不局限于public修饰的方法
getEnclosingClass()
Class 得到目标类所在的外围类的Class对象
getGenericInterfaces()
Type[] 得到目标类实现的接口对应的Type对象
getGenericSuperclass()
Type 得到目标类继承的父类所对应的Type对象
getInterfaces()
Class<?>[] 得到目标类实现的接口所对应的Class对象
getSuperclass()
Class 得到目标类继承的父类所对应的Class对象
isMemberClass()
boolean 目标类是否为成员类
cisAnonymousClass()
boolean 目标类是否为匿名类

 2.Field:

     我们知道一般类里包含有属性(成员)和方法(函数),竟然Class是描述类的信息,那么类其它部分应该会对应有描述它们的部分,而Field类型的对 象就是描述Class对象对应类的出现包括public、protected、private属性);一个Field对象对应描述一个类的属性;

    通过上文对Class的介绍,我们知道Class提供了四种接口函数可以得到对应属性的Field:

     1). getField(String name): 返回类型为Field,name为类中的属性名,得到的是描述类中的一个public属性对应的Field对象;如 Field mField =mClass.getField("mGestureID")  得到的是Gesture类中的一个public属性mGestureID对应的Field对象;

    2). getFields():返回类型为Field类型数组,得到的是描述类中的所有public属性对应的所有Field对象;

    3). getDeclaredField(String name):同getField(String name),只不过得到的Field对象描述的不只是public属性,

        还包括protected、private属性,也是说只要是在类中声明的属性;

    4). getDeclaredFields():getFields(),得到的是描述类中声明的所有属性(public、protected、private)对应的Field对象;

    Field类的相关函数接口介绍:

  Field类提供的相关接口介绍:(详细参看API)

接口 返回类型 接口功能实现
setAccessible(boolean flag) void 参数为true,只要是在类中声明的目标属性均可访问,为false,只有public目标属性可访问
set(Object object, Object value) void 给目标属性设置值(private、protected属性均不能访问,但可以通过先调用setAccessible(true)实现访问),第一个参数为目标属性所在类的对象,第二个参数为传入的值
get(Object object) Object 得到目标属性的值(private、protected属性均不能访问,但可以通过调用setAccessible(true)实现访问),参数为目标属性所在类的对象
setBoolean(Object object, boolean value) void 同set(Object object, Object value),只不过操作的数据类型为boolean
getBoolean(Object object) boolean 同get(Object object),只不过得到的数据类型为boolean
setByte(Object object, boolean value) void 同set(Object object, Object value),只不过操作的数据类型为byte
getByte(Object object) byte 同get(Object object),只不过得到的数据类型为byte
setShort(Object object, boolean value) void 同set(Object object, Object value),只不过操作的数据类型为short
getShort(Object object) short 同get(Object object),只不过得到的数据类型为short
setInt(Object object, boolean value) void 同set(Object object, Object value),只不过操作的数据类型为int
getInt(Object object) int 同get(Object object),只不过得到的数据类型为int
setLong(Object object, boolean value) void 同set(Object object, Object value),只不过操作的数据类型为long
getLong(Object object) long 同get(Object object),只不过得到的数据类型为long
setFloat(Object object, boolean value) void 同set(Object object, Object value),只不过操作的数据类型为float
getFloat(Object object) float 同get(Object object),只不过得到的数据类型为float
setDouble(Object object, boolean value) void 同set(Object object, Object value),只不过操作的数据类型为double
getDouble(Object object) double 同get(Object object),只不过得到的数据类型为double
setChar(Object object, boolean value) void 同set(Object object, Object value),只不过操作的数据类型为char
getChar(Object object) char 同get(Object object),只不过得到的数据类型为char
getName() String 得到目标属性的名字,不局限于private修饰符,只要是类中声明的属性
getGenericType() Type 得到目标属性的类型,不局限于private修饰符
getType() Class<?> 得到目标属性的类型对应的Class对象
getModifiers() int 得到目标属性的修饰符值(private为2、protected为4、public为1、static为8、final为16)
getDeclaringClass() Class<?> 得到目标属性所在类对应的Class对象


  下面就以一个示例代码来验证Field表中的函数接口的实现,如下:

     1). FieldBeReflected.java(被反射的类)

复制内容到剪贴板
  1. package com.stevenhu.field;    
  2.     
  3. public class FieldBeReflected     
  4. {    
  5.     private static String name;     
  6.     private static String name1;     
  7.     private boolean mBoolean = true;    
  8.     private final byte mByte = 111;    
  9.     private static final short mShort = 22;    
  10.     protected static int mInt;      
  11.     protected static long mLong;    
  12.     protected static float mFloat;    
  13.     protected static double mDouble;    
  14.     public static char mChar;    
  15.         
  16. }    

2). ReflectField.java(执行反射调用的类)

复制内容到剪贴板
  1. package com.stevenhu.reflection.test;    
  2. import java.lang.reflect.Field;    
  3. import java.lang.reflect.InvocationTargetException;    
  4. import java.lang.reflect.Method;    
  5. import java.lang.reflect.Type;    
  6.     
  7. import com.stevenhu.field.FieldBeReflected;    
  8.     
  9. public class ReflectField     
  10. {    
  11.     
  12.     public static void main(String[] args)     
  13.     {    
  14.         /*1.Class<?> clazz = Class.forName("com.stevenhu.field.FieldBeReflected");  
  15.          *2.FieldBeReflected mFieldBeReflected = new FieldBeReflected();  
  16.          *  Class<?> clazz = mFieldBeReflected.getClass();  
  17.          */    
  18.         Class<?> clazz = FieldBeReflected.class;    
  19.             
  20.         try {    
  21.             Field fName = clazz.getDeclaredField("name");    
  22.             Field fBoolean = clazz.getDeclaredField("mBoolean");    
  23.             Field fByte = clazz.getDeclaredField("mByte");    
  24.             Field fShort = clazz.getDeclaredField("mShort");    
  25.             Field fInt = clazz.getDeclaredField("mInt");    
  26.             Field fLong = clazz.getDeclaredField("mLong");    
  27.             Field fFloat = clazz.getDeclaredField("mFloat");    
  28.             Field fDouble = clazz.getDeclaredField("mDouble");    
  29.             Field fChar = clazz.getDeclaredField("mChar");    
  30.         
  31.             /*  
  32.              * 参数为true,只要是在类中声明的目标属性均可访问,  
  33.              * 为false,则被反射类和反射类在同一个包中时,private目标属性不可访问,  
  34.              * 不在同一个包中时,private、protected目标属性均不可访问  
  35.              */    
  36.             fName.setAccessible(true);    
  37.                 
  38.             /*给目标属性设置值(private属性不能访问,但可以通过先调用setAccessible(true)实现访问),  
  39.              * 由于ReflectField类中的name属性是静态的(static),所以方法的第一个实参传入的是  
  40.              * 目标属性所在类对应的Class对象clazz,也可以是类的实例clazz.newInstance();  
  41.              */    
  42.             fName.set(clazz, "reflection");    
  43.             //得到目标属性的值(private属性不能访问,但可以通过调用setAccessible(true)实现访问)    
  44.             String name = (String) fName.get(clazz);    
  45.             System.out.println(name);    
  46.                 
  47.             fBoolean.setAccessible(true);    
  48.             /*得到目标属性的布尔值,由于ReflectField类中的mBoolean属性是非静态的,  
  49.              * 所以此处的传入实参为目标属性所在类的实例clazz.newInstance()  
  50.              */    
  51.             boolean mBoolean = fBoolean.getBoolean(clazz.newInstance());    
  52.             System.out.println(mBoolean);    
  53.                 
  54.             fByte.setAccessible(true);    
  55.             //得到目标属性的Byte类型值    
  56.             byte mByte = fByte.getByte(clazz.newInstance());    
  57.             System.out.println(mByte);    
  58.                 
  59.             fShort.setAccessible(true);    
  60.             //得到目标属性的short整型值    
  61.             short mShort = fShort.getShort(clazz);    
  62.             System.out.println(mShort);    
  63.                 
  64.             fInt.setAccessible(true);    
  65.             //给目标属性设置整型值        
  66.             fInt.setInt(clazz, 222);    
  67.             //得到目标属性的整型值    
  68.             int mInt = fInt.getInt(clazz);    
  69.             System.out.println(mInt);    
  70.                 
  71.             fLong.setAccessible(true);    
  72.             //给目标属性设置Long整型值        
  73.             fLong.setLong(clazz, 2222);    
  74.             //得到目标属性的Long整型值    
  75.             Long mLong = fLong.getLong(clazz);    
  76.             System.out.println(mLong);    
  77.                 
  78.             fFloat.setAccessible(true);    
  79.             //给目标属性设置float类型值    
  80.             fFloat.setFloat(clazz, 22222);    
  81.             //得到目标属性的float类型值    
  82.             float mFloat = fFloat.getFloat(clazz);    
  83.             System.out.println(mFloat);    
  84.                 
  85.             fDouble.setAccessible(true);    
  86.             //给目标属性设置double类型值    
  87.             fDouble.setDouble(clazz, 222.222);    
  88.             //得到目标属性的double类型值    
  89.             double mDouble = fDouble.getDouble(clazz);    
  90.             System.out.println(mDouble);    
  91.                 
  92.             //给目标属性设置字符值(private、protected属性不能访问)    
  93.             fChar.setChar(clazz, 'a');    
  94.             //得到目标属性的字符值(private、protected属性不能访问)    
  95.             char mChar = fChar.getChar(clazz);    
  96.             System.out.println(mChar);    
  97.                 
  98.             //目标属性的名字,不局限于修饰符,只要是类中声明的属性    
  99.             String name1 = fName.getName();    
  100.             System.out.println(name1);    
  101.             //目标属性的类型,不局限于修饰符    
  102.             Type type = fName.getGenericType();    
  103.             System.out.println(type);    
  104.             //目标属性的类型对应的Class对象    
  105.             Class<?> clazz1 = fName.getType();        
  106.             System.out.println(clazz1);    
  107.             //目标属性所在类对应的Class对象    
  108.             Class<?> clazz2 = fName.getDeclaringClass();     
  109.             System.out.println(clazz2);    
  110.             //目标属性的权限修饰值(private为2、protected为4、public为1)    
  111.             int modifier = fName.getModifiers();    
  112.             int modifier1 = fByte.getModifiers();    
  113.             int modifier2 = fShort.getModifiers();    
  114.             System.out.println(modifier);    
  115.             System.out.println(modifier1);    
  116.             System.out.println(modifier2);    
  117.                 
  118.             System.out.println(fName.isAccessible());    
  119.             System.out.println(fChar.isAccessible());    
  120.                 
  121.         } catch (NoSuchFieldException e) {    
  122.             // TODO Auto-generated catch block    
  123.             e.printStackTrace();    
  124.         } catch (SecurityException e) {    
  125.             // TODO Auto-generated catch block    
  126.             e.printStackTrace();    
  127.         } catch (IllegalArgumentException e) {    
  128.             // TODO Auto-generated catch block    
  129.             e.printStackTrace();    
  130.         } catch (IllegalAccessException e) {    
  131.             // TODO Auto-generated catch block    
  132.             e.printStackTrace();    
  133.         } catch (InstantiationException e) {    
  134.             // TODO Auto-generated catch block    
  135.             e.printStackTrace();    
  136.         }     
  137.     }    
  138.     
  139. }    

 

3. Method:

       同Fiel一样,一个Method对象对应描述一个类的方法;

       Class对象也提供了四种接口函数得到对应方法的Method对象,如下:       

       1). getMethod(String name, Class<?>... parameterTypes):返回类型为Method,第一个参数name为类中的方法名,第二个参数为可变参数,传入的是参数类型对应的Class对象(方法的参数可能为多个的情况);该函数得到的是描述类中的一个public方法对应的Method对象;

      2). getMethods():返回类型为Method类型数组,得到的是描述类中的所有public方法对应的Method对象;

      3). Method getDeclaredMethod(String name, Class<?>...parameterTypes)

         同getMethod(String name, Class<?>... parameterTypes),只不过得到的Method对象描述的不只是public方法, 还包括

          protected、private方法,也是说只要是在类中声明的方法;

      4). getDeclaredMethods():getMethods(),得到的是描述类中声明的所有方法(public、protected、private)对应的FMethod对象;

   Method类的相关函数接口介绍:

   Method类提供的相关接口介绍:(详细参看API)

接口 返回类型 接口功能实现
setAccessible(boolean flag) void 参数为true,只要是在类中声明的目标方法均可访问,为false,只有public目标属性可访问
invoke(Object receiver, Object... args) Object 动态执行调用目标方法,第一个参数为Class对象或者类的实例,第二个参数为可变实参的对象(多个实参)
getDeclaringClass() Class<?> 得到目标方法所在类对应的Class对象
getExceptionTypes() Class<?> 得到目标方法抛出的异常类型对应的Class对象
getGenericExceptionTypes() Type[] 得到目标方法抛出的异常类型对应的Type对象
getReturnType() Class<?> 得到目标方法返回类型对应的Class对象
getGenericReturnType() Type 得到目标方法返回类型对应的Type对象
getParameterTypes() Class<?>[] 得到目标方法各参数类型对应的Class对象
getGenericParameterTypes() Type[] 得到目标方法各参数类型对应的Type对象
getModifiers() int 得到目标方法修饰符的值
getName() String 得到目标方法的名字


   下面就以一个示例代码来验证Method表中的函数接口的实现,如下:

   1). MethodBeReflected.java(被反射的类)

复制内容到剪贴板
  1. package com.stevenhu.method;    
  2.     
  3. public class MethodBeReflected    
  4. {    
  5.     
  6.     private static String mName;    
  7.     private static int mAge;    
  8.     private static float mWeight;    
  9.         
  10.     private String getmName()    
  11.     {    
  12.         return mName;    
  13.     }    
  14.     
  15.     protected void setmName(String mName)    
  16.     {    
  17.         this.mName = mName;    
  18.     }    
  19.     
  20.     protected static int getmAge()    
  21.     {    
  22.         return mAge;    
  23.     }    
  24.     
  25.     private static void setmAge(int age)     
  26.     {    
  27.         mAge = age;    
  28.     }    
  29.     
  30.     private float getmWeight() throws Exception, NoSuchMethodException, SecurityException    
  31.     {    
  32.         return mWeight;    
  33.     }    
  34.     
  35.     protected void setmWeight(float mWeight)     
  36.     {    
  37.         this.mWeight = mWeight;    
  38.     }    
  39.     
  40.     private void setAllValues(String name, int age, float weight)    
  41.     {    
  42.         this.mName = name;    
  43.         this.mAge = age;    
  44.         this.mWeight = weight;    
  45.     }    
  46. }    

2)ReflectMethod.java(执行反射的类)

复制内容到剪贴板
  1. package com.stevenhu.reflection.test;    
  2.     
  3. import java.lang.reflect.InvocationTargetException;    
  4. import java.lang.reflect.Method;    
  5. import java.lang.reflect.Type;    
  6.     
  7. import com.stevenhu.method.MethodBeReflected;    
  8.     
  9. public class ReflectMethod     
  10. {    
  11.         
  12.     /**  
  13.      * @param args  
  14.      */    
  15.     public static void main(String[] args)     
  16.     {    
  17.         // TODO Auto-generated method stub    
  18.     
  19.         Class<?> clazz = MethodBeReflected.class;    
  20.             
  21.         try     
  22.         {    
  23.             //第一个实参为方法名,第二个实参为方法参数类型对应的class对象    
  24.             Method nameMethod = clazz.getDeclaredMethod("setmName", String.class);    
  25.             Method ageMethod = clazz.getDeclaredMethod("setmAge"int.class);    
  26.             Method weightMethod = clazz.getDeclaredMethod("setmWeight"float.class);    
  27.             Method allValuesMethod = clazz.getDeclaredMethod("setAllValues"new Class[]{String.classint.classfloat.class});    
  28.                 
  29.             nameMethod.setAccessible(true);    
  30.             //调用setmName方法,给ReflectMethod类中的属性mName赋值为"stevenhu"    
  31.             nameMethod.invoke(clazz.newInstance(), "lisa");         
  32.             nameMethod = clazz.getDeclaredMethod("getmName"null);    
  33.             nameMethod.setAccessible(true);    
  34.             //调用getmName方法,得到mName的值    
  35.             String name1 = (String) nameMethod.invoke(clazz.newInstance(), null);    
  36.             System.out.println(name1);    
  37.                 
  38.             ageMethod.setAccessible(true);    
  39.             /*调用setmAge方法设置年龄,由于该方法是静态方法,所以第一个实参可为类的Class对象clazz,  
  40.              * 也可以是类的对象clazz.newInstance();  
  41.              */    
  42.             ageMethod.invoke(clazz, 21);    
  43.             ageMethod = clazz.getDeclaredMethod("getmAge"null);    
  44.             ageMethod.setAccessible(true);    
  45.             //调用getmAge方法,得到之前设置的年龄    
  46.             int age1 = (Integer) ageMethod.invoke(clazz, null);     
  47.             System.out.println(age1);    
  48.                 
  49.             weightMethod.setAccessible(true);    
  50.             //调用setmWeight方法,设置体重    
  51.             weightMethod.invoke(clazz.newInstance(), new Float(50.5));    
  52.             weightMethod = clazz.getDeclaredMethod("getmWeight",null);    
  53.             weightMethod.setAccessible(true);    
  54.             //调用getmWeight方法,得到之前设置的体龄    
  55.             float weight1 = (Float) weightMethod.invoke(clazz.newInstance(), null);     
  56.             System.out.println(weight1);    
  57.                 
  58.             allValuesMethod.setAccessible(true);    
  59.             /*调用ReflectMethod的setAllValues方法赋值  
  60.              * 注:此处不能直接传入实参63.5;浮点型必须创建Float对象  
  61.              * 整型和字符串可创建Integer、String对象,也可以不创建  
  62.              */    
  63.             //allValuesMethod.invoke(clazz.newInstance(), new String("stevenhu"), new Integer(23), new Float(63.5));    
  64.             allValuesMethod.invoke(clazz.newInstance(), "stevenhu"23new Float(63.5));    
  65.                 
  66.             nameMethod = clazz.getDeclaredMethod("getmName"null);    
  67.             nameMethod.setAccessible(true);    
  68.             String name2 = (String) nameMethod.invoke(clazz.newInstance(), null);    
  69.             System.out.println(name2);    
  70.                 
  71.             ageMethod = clazz.getDeclaredMethod("getmAge"null);    
  72.             ageMethod.setAccessible(true);    
  73.             int age2 = (Integer) ageMethod.invoke(clazz.newInstance(), null);           
  74.             System.out.println(age2);    
  75.                 
  76.             weightMethod = clazz.getDeclaredMethod("getmWeight"null);    
  77.             weightMethod.setAccessible(true);    
  78.             float weight2 = (Float) weightMethod.invoke(clazz.newInstance(), null);         
  79.             System.out.println(weight2);    
  80.                 
  81.             //得到目标方法所在类对应的Class对象    
  82.             Class<?> clazz1 = weightMethod.getDeclaringClass();    
  83.                 
  84.             //得到目标方法抛出的异常类型对应的Class对象    
  85.             Class<?>[] clazzs1 = weightMethod.getExceptionTypes();    
  86.             for (Class cl : clazzs1)    
  87.             {    
  88.                 System.out.println(cl);    
  89.             }               
  90.             //得到目标方法抛出的异常类型对应的Type对象    
  91.             Type[] types1 = weightMethod.getGenericExceptionTypes();    
  92.             //得到目标方法返回类型对应的Class对象    
  93.             Class<?> clazz2 = nameMethod.getReturnType();    
  94.             //得到目标方法返回类型对应的Type对象    
  95.             Type type = nameMethod.getGenericReturnType();    
  96.             //得到目标方法各参数类型对应的Class对象    
  97.             Class<?>[] clazzs2 = allValuesMethod.getParameterTypes();    
  98.             //得到目标方法各参数类型对应的Type对象    
  99.             Type[] types2 = allValuesMethod.getGenericParameterTypes();    
  100.             //得到目标方法修饰符的值    
  101.             int modifier = ageMethod.getModifiers();    
  102.             System.out.println(modifier);    
  103.             //得到目标方法的名字    
  104.             String methodName = nameMethod.getName();    
  105.             System.out.println(nameMethod.isVarArgs());    
  106.                 
  107.         } catch (NoSuchMethodException e)     
  108.         {    
  109.             // TODO Auto-generated catch block    
  110.             e.printStackTrace();    
  111.         } catch (SecurityException e)     
  112.         {    
  113.             // TODO Auto-generated catch block    
  114.             e.printStackTrace();    
  115.         } catch (IllegalAccessException e) {    
  116.             // TODO Auto-generated catch block    
  117.             e.printStackTrace();    
  118.         } catch (IllegalArgumentException e) {    
  119.             // TODO Auto-generated catch block    
  120.             e.printStackTrace();    
  121.         } catch (InvocationTargetException e) {    
  122.             // TODO Auto-generated catch block    
  123.             e.printStackTrace();    
  124.         } catch (InstantiationException e) {    
  125.             // TODO Auto-generated catch block    
  126.             e.printStackTrace();    
  127.         }    
  128.     }    
  129.     
  130. }    
复制内容到剪贴板
  1. import java.lang.reflect.Field;    
  2. import java.lang.reflect.Method;    
  3. import java.text.SimpleDateFormat;    
  4. import java.util.Date;    
  5.     
  6. public class TestReflect {    
  7.     public String name;    
  8.     
  9.     private int age;    
  10.     
  11.     public String getName() {    
  12.         return name;    
  13.     }    
  14.     
  15.     public void setName(String name) {    
  16.         this.name = name;    
  17.     }    
  18.     
  19.     public String sayHello() {    
  20.         return "hello";    
  21.     }    
  22.     
  23.     public String sayHello(String name) {    
  24.         return name + ", hello";    
  25.     }    
  26.     
  27.     public String sayHello(String name, Date date) {    
  28.         return name + ", hello at " + new SimpleDateFormat("yyyy-MM-dd HH:mm:ss").format(date);    
  29.     }    
  30.     
  31.     @Override    
  32.     public String toString() {    
  33.         return "the object: name=" + name + ";age=" + age;    
  34.     }    
  35.     
  36.     /**  
  37.      * @param args  
  38.      */    
  39.     public static void main(String[] args) {    
  40.         try {    
  41.     
  42.             // 测试String的生成    
  43.     
  44.             String testStr = "testString";    
  45.             Class getClassWay1 = testStr.getClass();    
  46.             Class getClassWay2 = String.class;    
  47.             Class getClassWay3 = Class.forName("java.lang.String");    
  48.             System.out.println(getClassWay1 == getClassWay2);    
  49.             System.out.println(getClassWay2 == getClassWay3);    
  50.     
  51.             String str1 = (String) Class.forName("java.lang.String").newInstance();    
  52.             System.out.println(str1);    
  53.     
  54.             String str2 = (String) Class.forName("java.lang.String"// 得到字节码即事例对象    
  55.                     .getConstructor(StringBuffer.class// 获得该对象中的特定构造器    
  56.                     .newInstance(new StringBuffer("abc")); // 根据构造器产生对象实例    
  57.             System.out.println(str2.charAt(1));    
  58.     
  59.             System.out.println("==========");    
  60.             /////////////////////////////////////////////////////////    
  61.             Class<?> clazz = Class.forName("com.huabao.test.Demo");    
  62.             Object obj = clazz.newInstance();    
  63.             System.out.println(obj);    
  64.     
  65.             Field fname = clazz.getField("name");    
  66.             fname.set(obj, "java");    
  67.             Field fage = clazz.getDeclaredField("age");    
  68.             fage.setAccessible(true);    
  69.             fage.setInt(obj, 100);    
  70.             System.out.println(obj);    
  71.             System.out.println(fage.getInt(obj));    
  72.     
  73.             System.out.println("==========");    
  74.     
  75.             Method m0 = clazz.getMethod("getName"null);    
  76.             System.out.println(m0.invoke(obj, null));    
  77.             Method m00 = clazz.getMethod("setName"new Class[] { String.class });    
  78.             System.out.println(m00.invoke(obj, new Object[] { "java111" }));    
  79.             System.out.println(obj);    
  80.     
  81.             Method m1 = clazz.getMethod("sayHello"new Class[0]);    
  82.             System.out.println(m1.invoke(obj, new Object[0]).toString());    
  83.     
  84.             Method m2 = clazz.getMethod("sayHello"new Class[] { String.class });    
  85.             System.out.println(m2.invoke(obj, new Object[] { "java world" }).toString());    
  86.     
  87.             Method m3 = clazz.getMethod("sayHello"new Class[] { String.class, Date.class });    
  88.             System.out.println(m3.invoke(obj, new Object[] { "java world"new Date() }).toString());    
  89.         } catch (Exception e) {    
  90.             e.printStackTrace();    
  91.         }    
  92.     }    
  93. }    
复制内容到剪贴板
  1. import java.util.HashMap;    
  2.     
  3. public class TestClass {    
  4.     
  5.     public static void main(String[] args) {    
  6.     
  7.         Object a = new HashMap();    
  8.         System.out.println(a.getClass().isArray());    
  9.         System.out.println(a.getClass().getGenericSuperclass());    
  10.         System.out.println(a.getClass().getPackage());    
  11.         System.out.println(a.getClass().getSimpleName());    
  12.     
  13.         System.out.println("+++");    
  14.         for (Class c : a.getClass().getInterfaces()) {    
  15.             System.out.println(c.getName());    
  16.         }    
  17.         System.out.println("+++");    
  18.         System.out.println(a.getClass().getComponentType());    
  19.         System.out.println(a.getClass().getModifiers());    
  20.         System.out.println(a.getClass().isLocalClass());    
  21.         System.out.println(a.getClass().isLocalClass());    
  22.     }    
  23.     
  24. }    

JDK源码解析

 

Class

该类主要获得编译后的字节码信息,主要包含类的加载/构造方法/普通方法/属性列表/包信息/父类信息的主要信息        
    
        Class类(在java.lang包中,Instances of the class Classrepresent classes and interfaces in a running Javaapplication):
        在Java中,每个class都有一个相应的Class对象。也就是说,当我们编写一个类,编译完成后,在生成的.class文件中,就会产生一个Class对象,用于表示这个类的类型信息
        获取Class实例的三种方式:
         (1)利用对象调用getClass()方法获取该对象的Class实例;
         (2)使用Class类的静态方法forName(),用类的名字获取一个Class实例(staticClass forName(String className)  Returns the Classobject associated with the class or interface with the given stringname. );
         (3)运用.class的方式来获取Class实例,对于基本数据类型的封装类,还可以采用.TYPE来获取相对应的基本数据类型的Class实例
        在newInstance()调用类中缺省的构造方法 ObjectnewInstance()(可在不知该类的名字的时候,常见这个类的实例) Creates a new instance of the class represented by this Classobject.
        在运行期间,如果我们要产生某个类的对象,Java虚拟机(JVM)会检查该类型的Class对象是否已被加载。如果没有被加载,JVM会根据类的名称 找到.class文件并加载它。一旦某个类型的Class对象已被加载到内存,就可以用它来产生该类型的所有对象

复制内容到剪贴板
  1. public class ClassTest {  
  2.       public static void main(String [] args)throws Exception{  
  3.           String str1="abc";  
  4.           Class cls1=str1.getClass();  
  5.           Class cls2=String.class;  
  6.           Class cls3=Class.forName("java.lang.String");  
  7.           System.out.println(cls1==cls2);  
  8.           System.out.println(cls1==cls3);  
  9.       }  
  10.   }  

         返回结果为:true,true.
            解释:虚拟机只会产生一份字节码, 用这份字节码可以产生多个实例对象。

 

Method  Field

Method 提供关于类或接口上单独某个方法(以及如何访问该方法)的信息。所反映的方法可能是类方法或实例方法(包括抽象方法)。

Method 允许在匹配要调用的实参与底层方法的形参时进行扩展转换;但如果要进行收缩转换,则会抛出 IllegalArgumentException

 

Field 提供有关类或接口的单个字段的信息,以及对它的动态访问权限。反射的字段可能是一个类(静态)字段或实例字段。

Array 允许在执行 get 或 set 访问操作期间进行扩展转换,但如果将发生收缩转换,则抛出一个 IllegalArgumentException

感谢 xinyuan_java 支持 磐实编程网 原文地址:
blog.csdn.net/xinyuan_java/article/details/51180959

文章信息

发布时间:2016-04-18

作者:xinyuan_java

发布者:aquwcw

浏览次数: