反射私有方法

任务概述:今天黄俊兄发现把构造方法私有化之后,Spring一样可以把Bean实例创建出来,于是乎,找了一下午的答案,在老张的提点下,得出了结果,还是用的反射,以下是总结.

1.私有构造函数的调用:
主要的类:java.lang.reflect.Constructor

首先要拿到构造方法,具体如下:
Class clazz = Class.forName("Bean");
Constructor[] cons = clazz.getDeclaredConstructors();
这样拿到的cons数组保存着所有的这个类的构造函数,包括私有的和公有的。
迭代并调用con.toGenericString()可以拿到每个构造方法的全限定名(带修饰符,带参数)。
如果要拿单个构造方法,可以调用getDeclaredConstructor(Class args);来拿,只要参数匹配了就可以得到,如果没有参数,则传null即可。

然后需要调用Constructor的方法setAccessible设置为true,表示允许使用私有构造函数。这个方法是父类AccessibleObject的方法,这个类的自类有Constructor, Field, Method分别代表构造函数,成员变量,和成员方法。也就是说使用这个方法,可以访问类的所有的方法、变量、构造函数。

最后,使用Constructor的newInstance方法就可以把对象创建出来了:
Bean bean = (Bean)con.newInstance(Object args);
没有参数的构造函数可以传null进去。

2.调用私有的方法:
主要的类:java.lang.reflect.Method

首先还是要拿到那个私有的方法,可以调用
Method[] ms = clazz.getDeclaredMethods();来拿到所有的方法的集合。
也可以拿某个单独的方法:getDeclaredMethod(String name, Class... parameterTypes)匹配名字和参数的方法会被拿到,方法若没有参数传null.
同样可以调用toGenericString()看全限定名。

然后同样需要调用setAccessible(true)来设置可以方法私有方法。

最后,就可以使用method.invoke(bean, Object args);来调用这个方法。两个参数分别是对象,和这个方法的参数。

3.调用私有成员变量,也叫字段。
主要类:java.lang.reflect.Field

首先还是要拿到要使用的字段:
还是两种,拿全部:Field[] fs = clazz.getDeclaredFields();
拿某个:clazz.getDeclaredField(String fieldName);

老步骤:setAccessible(true)

最后, f.get(obj) ,就能拿到传入对象的私有变量。

测试:

MyBean.java :

package cn.yzj.entities;

public class MyBean
{
private String name;
private Integer age;

private MyBean()
{
this.name ="ABC";
}

private MyBean(String name,Integer age)
{
this.name = name;
this.age = age;
}

public String getName()
{
return name;
}
public void setName(String name)
{
this.name = name;
}

private void printName()
{
System.out.println(this.name);
}
}



Test1.java:

package cn.yzj;

import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;

import cn.yzj.entities.MyBean;

public class Test1
{
public static void main(String[] args) throws ClassNotFoundException,
IllegalArgumentException, InstantiationException,
IllegalAccessException, InvocationTargetException,
SecurityException, NoSuchMethodException
{
Class clazz = Class.forName("cn.yzj.entities.MyBean");

// 获得私有成员变量并返回.
// getField(clazz);

// 调用私有方法
// getMethod(clazz);

// 用私有构造方法构造对象并返回.
// getConstrutor(clazz);

}


//用私有构造方法构建实例的方法
private static MyBean getConstrutor(Class clazz)
throws NoSuchMethodException, InstantiationException,
IllegalAccessException, InvocationTargetException
{

// 拿所有的构造方法.
/*
* Constructor[] cons = clazz.getDeclaredConstructors(); for(Constructor
* con : cons) { System.out.println(con.toGenericString()); }
*/

// 1.拿无参数的构造方法.
Constructor con = clazz.getDeclaredConstructor(null);

// 2.设置可以访问.
con.setAccessible(true);

// 3.创建实例
MyBean myBean = (MyBean) con.newInstance(null);
return myBean;
}

private static Object getField(Class clazz) throws NoSuchMethodException,
InstantiationException, IllegalAccessException,
InvocationTargetException
{
MyBean myBean = getConstrutor(clazz);

Field[] fs = clazz.getDeclaredFields();

for (Field f : fs)
{
System.out.println(f.toGenericString());
}

fs[0].setAccessible(true);

return fs[0].get(myBean);
}

private static void getMethod(Class clazz) throws NoSuchMethodException,
InstantiationException, IllegalAccessException,
InvocationTargetException
{
MyBean myBean = getConstrutor(clazz);

Method[] ms = clazz.getDeclaredMethods();

for (Method m : ms)
{
System.out.println(m.toGenericString());
}

ms[0].setAccessible(true);

ms[0].invoke(myBean, null);
}

}


原文在http://groups.csdn.net/itcast_6/topic/8e48d424-be3a-48cf-9708-e09866ac695d.aspx

你可能感兴趣的:(反射私有方法)