java 反射
一:
java.lang.Class类:Class是一个类,构造方法是private,由JVM创建(无法直接new CLass(),其对象是内存里的一份字节码)。
放射: 反射是java语言的一个特性,它允程序在运行时(注意不是编译的时候)来进行自我检查并且对内部的成员进行操作。
例如它允许一个java的类获取他所有的成员变量和方法并且显示出来。
Java 的这一能力在实际应用中也许用得不是很多,但是在其它的程序设计语言中根本就不存在这一特性。
例如,Pascal、C 或者 C++ 中就没有办法在程序中获得函数定义相关的信息。
reflection可以动态的载入并取得 Java 组件(类) 的属性。(来自Sun)
Class 类的实例表示正在运行的 Java 应用程序中的类和接口。枚举是一种类,注释是一种接口。每个数组属于被映射为 Class 对象的一个类,所有具有相同元素类型和维数的数组都共享该 Class 对象。
基本的 Java类型(boolean、byte、char、short、int、long、float 和 double)和关键字 void 也表示为 Class 对象。Class 没有公共构造方法。
如何得到类的信息: Class 对象是在加载类时由 Java 虚拟机以及通过调用类加载器中的 defineClass 方法自动构造的。如下,三种方法可以得到类的信息:
可以通过类名称(Date.class);可以通过对象得到(p1.getclass);也可以通过一个运行时的字符串(Class.forname("XXXXX"))得到。
Person p1 = new Person(); //下面的这三种方式都可以得到字节码 Class c1 = Date.class; Class<? extends Person> c2 = p1.getClass(); Class<?> c3 = null; //若存在则加载,否则新建,往往使用第三种,类的名字在写源程序时不需要知道,到运行时再传递过来 try { c3 = Class.forName("java.lang.String");
// forName中的参数一定是完整的类名(包名+类名),并且这个方法需要捕获异常 } catch (ClassNotFoundException e) { e.printStackTrace(); }
下面有一个例子:
Class c = null; try { c = Class.forName("reflection.Person"); Method m[] = c.getDeclaredMethods(); for (int i = 0; i < m.length; i++) { System.out.println(m[i].toString()); } } catch (ClassNotFoundException e) { e.printStackTrace(); }
输出:如下,输出类的各方法名,以及它们的限制符,和返回类型和参数
public void reflection.Person.setName(java.lang.String)
public void reflection.Person.setAge(int)
public int reflection.Person.getAge()
private java.lang.String reflection.Person.func(java.lang.String)
例2: 如下利用Class的newInstance方法相当于调用类的默认的构造器。如果类没有无参的构造器,就会抛出异常。
Person p = new Person(5, ""); Class<? extends Person> c = p.getClass(); try { Person da = c.newInstance(); } catch (InstantiationException e) { e.printStackTrace(); } catch (IllegalAccessException e) { e.printStackTrace(); }
二:常用方法
1 isPrimitive(判断是否是基本类型的字节码)
2 java中构造方法没有先后顺序,通过类型和参数个数区分。
java中构造方法没有先后顺序,通过类型和参数个数区分。
try { Constructor<Person> con1 = Person.class.getConstructor(int.class, String.class); Constructor<Person> con2 = Person.class.getConstructor(); } catch (NoSuchMethodException e) { e.printStackTrace(); }
3 Filed类代表某一类中的一个成员变量。
Field[] f = Person.class.getFields(); // 只能得到public的
Field f1 = Person.class.getField("age"); // 只能得到public的
Field[] f2 = Person.class.getDeclaredFields(); // 各种可见性的都可以得到,但是继承的得不到
Field f3 = Person.class.getDeclaredField("age"); //各种可见性的都可以得到,但是继承的得不到
Person p = new Person(5, "jack"); Field f3 = Person.class.getDeclaredField("age"); f3.setAccessible(true); // 如果是私有的属性, 一定要临时改其属性 String rs = (String) f3.get(p);
4: 给定一个类的全名,调用main方法:
Method m = Class.forName(str).getMethod("main",String[].class);
m.invoke(null, new Object[]{new String[]{"111","222","333"}});
m.invoke(null, (Object)new String[]{"111","222","333"});//这个可以说明,数组也是Object
5: 模拟instanceof 操作
Person p = new Person(5, "jack");
Person.class.isInstance(p);
三:Method类
try { String str = "shfsfs"; //包开头是com表示是sun内部用的,java打头的才是用户的 Method mtCharAt = String.class.getMethod("charAt", int.class); Object ch = mtCharAt.invoke(str, 1);//若第一个参数是null,则肯定是静态方法 System.out.println(ch); System.out.println(mtCharAt.invoke(str, new Object[] {2}));//1.4语法 } catch (NoSuchMethodException e) { e.printStackTrace(); } catch (InvocationTargetException e) { e.printStackTrace(); }