您的位置:首页 > 编程语言 > Java开发

Java反射(一)

2015-06-04 21:28 459 查看
Java的反射机制可以在运行时再装配代码,使程序的编写更加灵活.(高大上,看起来就很厉害的样子,然而到底有什么用呢?)
Java的反射机制可以访问一个未知对象的属性,方法,构造方法(有参or无参),
实现Java的反射共有三种方法:


1.通过forName()方法;

2.x.class;

3.x.getClass()。



我们主要掌握第一种.
下面来个Demo:

public static void main(String[] args) throws Exception {
reflectTest();
reflectTest2("Tom","Cat");
reflectTest3("Ben","Dog");
reflectTest4();
}
public static void reflectTest() throws Exception {
System.out.println("1");
Class tclass2 = Class.forName("reflect.bean.Pet");
Object obj = tclass2.newInstance();// 调用非静态new 一个对象,调用静态方法则不需要

}

public static void reflectTest2(String name, String kind) throws Exception {
System.out.println("2");
Class tclass = Class.forName("reflect.bean.Pet");
Constructor con = tclass.getConstructor(String.class, String.class);
con.newInstance(name, kind);

}
public static void reflectTest3(String name, String kind) throws Exception{
System.out.println("3");
Class tclass = Class.forName("reflect.bean.Pet");
Object obj =tclass.newInstance();
Method met=tclass.getMethod("PetInfo", String.class,String.class);
met.invoke(obj, name,kind);
}
public static void reflectTest4() throws Exception {
System.out.println("4");
Class tclass2 = Class.forName("reflect.bean.Pet");
Object obj = tclass2.newInstance();
Method met =tclass2.getMethod("PetInfo", null);
met.invoke(obj, null);
}
Pet类

package reflect.bean;

public class Pet {
private String kind;
private String name;

public Pet(String kind, String name) {
super();
this.kind = kind;
this.name = name;
System.out.println("Pet---" + kind + "---" + name);
}

public Pet() {
super();
System.out.println("Pet");
}

public void PetInfo() {
System.out.println("tom---Cat");
}

public void PetInfo(String kind, String name) {
System.out.println(name + "---" + kind);
}
}
输出结果:

1
Pet
2
Pet---Tom---Cat
3
Pet
Dog---Ben
4
Pet
tom---Cat


我们来看看Class.forName()方法:

public static Class<?> forName(String className)
throws ClassNotFoundException {

return forName0(className, true, ClassLoader.getCallerClassLoader());
}


Class.forName()的主要任务就是把指定路径的类装载到jvm中.

static ClassLoader getCallerClassLoader() {
// NOTE use of more generic Reflection.getCallerClass()
Class caller = Reflection.getCallerClass(3);
// This can be null if the VM is requesting it
if (caller == null) {
return null;
}
// Circumvent security check since this is package-private
return caller.getClassLoader0();
}


这里判断了对象的访问权限,如果访问对象修饰符为 private,需要禁止Field的访问控制检查,, boolean)]setAccessible设置为true.

static void
, boolean)]setAccessible(AccessibleObject[] array,
boolean flag)


使用单一安全性检查(为了提高效率)为一组对象设置
accessible
标志的便捷方法。
然后,newInstance(),代码太长,可以自己去看看.要使用这个方法必须在对应类加载并连接的状态下,而前面的Class.forName()就保证了newInstance的使用满足条件.(正是在这里调用Class的方法,从而获得了更好的灵活性.)
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: