使用java反射操作类的构造函数,成员变量和成员方法
2008-02-25 11:09
886 查看
在java.lang.reflect包中有三个类Field,Method,Constructor.分别描述域,方法,构造器。参考API,关于这三个类的说明。
在运行时使用反射分析对象,如果访问的是私有域或是私有方法,私有构造函数,会抛出IllegalAccessException.因为反射机制的默认行为受限于java的访问控制,然而,如果一个java程序没有受到安全管理器的控制,就可以覆盖访问控制。为了达到这个目的,需要调用Field,Method,Constructor对象的setAccessible方法,setAccessible方法是AccessibleObject类的一个方法,是Field,Method,Constructor类的公共超类。
setAccessible(true);
不要过多地使用反射:破坏类封装;效率偏低;编译器不容易帮助发现错误,在运行时才会被发现。
我使用反射,是为了写单元测试代码,对运行对象的分析,改变对象的私有域等,达到测试需要的覆盖率。
以下提供代码事例:
测试类:
public class RunBing ...{
private String name;
private int age;
public String sex;
public RunBing() ...{
}
private RunBing(String name) ...{
this.name = name;
}
public void setName(String name) ...{
this.name = name;
}
public void setAge(int age) ...{
this.age = age;
}
public String getName() ...{
return name;
}
public int getAge() ...{
return age;
}
public String run() ...{
return "run method";
}
private String add() ...{
return "add method";
}
}
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.lang.reflect.Constructor;
public class ReflectHelp ...{
/** *//**
* 通过构造器取得实例
* @param className 类的全限定名
* @param intArgsClass 构造函数的参数类型
* @param intArgs 构造函数的参数值
*
* @return Object
*/
public static Object getObjectByConstructor(String className,Class[] intArgsClass,Object[] intArgs)...{
Object returnObj= null;
try ...{
Class classType = Class.forName(className);
Constructor constructor = classType.getDeclaredConstructor(intArgsClass); //找到指定的构造方法
constructor.setAccessible(true);//设置安全检查,访问私有构造函数必须
returnObj = constructor.newInstance(intArgs);
} catch (NoSuchMethodException ex) ...{
ex.printStackTrace();
} catch (Exception ex) ...{
ex.printStackTrace();
}
return returnObj;
}
/** *//**
* 修改成员变量的值
* @param Object 修改对象
* @param filedName 指定成员变量名
* @param filedValue 修改的值
*
* @return
*/
public static void modifyFileValue(Object object, String filedName,
String filedValue) ...{
Class classType = object.getClass();
Field fild = null;
try ...{
fild = classType.getDeclaredField(filedName);
fild.setAccessible(true);//设置安全检查,访问私有成员变量必须
fild.set(object, filedValue);
} catch (NoSuchFieldException ex) ...{
ex.printStackTrace();
} catch (Exception ex) ...{
ex.printStackTrace();
}
}
/** *//**
* 访问类成员变量
* @param Object 访问对象
* @param filedName 指定成员变量名
* @return Object 取得的成员变量的值
* */
public static Object getFileValue(Object object, String filedName) ...{
Class classType = object.getClass();
Field fild = null;
Object fildValue = null;
try ...{
fild = classType.getDeclaredField(filedName);
fild.setAccessible(true);//设置安全检查,访问私有成员变量必须
fildValue = fild.get(object);
} catch (NoSuchFieldException ex) ...{
ex.printStackTrace();
} catch (Exception ex) ...{
ex.printStackTrace();
}
return fildValue;
}
/** *//**
* 调用类的方法,包括私有
* @param Object 访问对象
* @param methodName 指定成员变量名
* @param type 方法参数类型
* @param value 方法参数指
* @return Object 方法的返回结果对象
* */
public static Object useMethod(Object object, String methodName,
Class[] type, Class[] value) ...{
Class classType = object.getClass();
Method method = null;
Object fildValue = null;
try ...{
method = classType.getDeclaredMethod(methodName, type);
method.setAccessible(true);//设置安全检查,访问私有成员方法必须
fildValue = method.invoke(object, value);
} catch (NoSuchMethodException ex) ...{
ex.printStackTrace();
} catch (Exception ex) ...{
ex.printStackTrace();
}
return fildValue;
}
public static void main(String[] args) ...{
RunBing runbing = new RunBing();
runbing.setName("aaa");
//访问成员变量
String value = (String) ReflectHelp.getFileValue(runbing, "sex");
System.out.println("value:" + value);
//修改成员变量
ReflectHelp.modifyFileValue(runbing, "sex", "bbb");
System.out.println("value:" + runbing.getName());
//检查修改后的变量值
value = (String) ReflectHelp.getFileValue(runbing, "sex");
System.out.println("value:" + value);
//调用方法
value = (String) ReflectHelp.useMethod(runbing, "add", new Class[] ...{},
new Class[] ...{});
System.out.println("value:" + value);
//使用指定构造函数
Class[] inArgs = new Class[]...{String.class};
Object[] inArgsParms = new Object[]...{"hanj"};
RunBing runBing2 = (RunBing)ReflectHelp.getObjectByConstructor("com.ccit.hj.reflect.RunBing",inArgs,inArgsParms);
value = (String) ReflectHelp.getFileValue(runBing2, "name");
System.out.println("cc -- value:" + value);
}
在运行时使用反射分析对象,如果访问的是私有域或是私有方法,私有构造函数,会抛出IllegalAccessException.因为反射机制的默认行为受限于java的访问控制,然而,如果一个java程序没有受到安全管理器的控制,就可以覆盖访问控制。为了达到这个目的,需要调用Field,Method,Constructor对象的setAccessible方法,setAccessible方法是AccessibleObject类的一个方法,是Field,Method,Constructor类的公共超类。
setAccessible(true);
不要过多地使用反射:破坏类封装;效率偏低;编译器不容易帮助发现错误,在运行时才会被发现。
我使用反射,是为了写单元测试代码,对运行对象的分析,改变对象的私有域等,达到测试需要的覆盖率。
以下提供代码事例:
测试类:
public class RunBing ...{
private String name;
private int age;
public String sex;
public RunBing() ...{
}
private RunBing(String name) ...{
this.name = name;
}
public void setName(String name) ...{
this.name = name;
}
public void setAge(int age) ...{
this.age = age;
}
public String getName() ...{
return name;
}
public int getAge() ...{
return age;
}
public String run() ...{
return "run method";
}
private String add() ...{
return "add method";
}
}
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.lang.reflect.Constructor;
public class ReflectHelp ...{
/** *//**
* 通过构造器取得实例
* @param className 类的全限定名
* @param intArgsClass 构造函数的参数类型
* @param intArgs 构造函数的参数值
*
* @return Object
*/
public static Object getObjectByConstructor(String className,Class[] intArgsClass,Object[] intArgs)...{
Object returnObj= null;
try ...{
Class classType = Class.forName(className);
Constructor constructor = classType.getDeclaredConstructor(intArgsClass); //找到指定的构造方法
constructor.setAccessible(true);//设置安全检查,访问私有构造函数必须
returnObj = constructor.newInstance(intArgs);
} catch (NoSuchMethodException ex) ...{
ex.printStackTrace();
} catch (Exception ex) ...{
ex.printStackTrace();
}
return returnObj;
}
/** *//**
* 修改成员变量的值
* @param Object 修改对象
* @param filedName 指定成员变量名
* @param filedValue 修改的值
*
* @return
*/
public static void modifyFileValue(Object object, String filedName,
String filedValue) ...{
Class classType = object.getClass();
Field fild = null;
try ...{
fild = classType.getDeclaredField(filedName);
fild.setAccessible(true);//设置安全检查,访问私有成员变量必须
fild.set(object, filedValue);
} catch (NoSuchFieldException ex) ...{
ex.printStackTrace();
} catch (Exception ex) ...{
ex.printStackTrace();
}
}
/** *//**
* 访问类成员变量
* @param Object 访问对象
* @param filedName 指定成员变量名
* @return Object 取得的成员变量的值
* */
public static Object getFileValue(Object object, String filedName) ...{
Class classType = object.getClass();
Field fild = null;
Object fildValue = null;
try ...{
fild = classType.getDeclaredField(filedName);
fild.setAccessible(true);//设置安全检查,访问私有成员变量必须
fildValue = fild.get(object);
} catch (NoSuchFieldException ex) ...{
ex.printStackTrace();
} catch (Exception ex) ...{
ex.printStackTrace();
}
return fildValue;
}
/** *//**
* 调用类的方法,包括私有
* @param Object 访问对象
* @param methodName 指定成员变量名
* @param type 方法参数类型
* @param value 方法参数指
* @return Object 方法的返回结果对象
* */
public static Object useMethod(Object object, String methodName,
Class[] type, Class[] value) ...{
Class classType = object.getClass();
Method method = null;
Object fildValue = null;
try ...{
method = classType.getDeclaredMethod(methodName, type);
method.setAccessible(true);//设置安全检查,访问私有成员方法必须
fildValue = method.invoke(object, value);
} catch (NoSuchMethodException ex) ...{
ex.printStackTrace();
} catch (Exception ex) ...{
ex.printStackTrace();
}
return fildValue;
}
public static void main(String[] args) ...{
RunBing runbing = new RunBing();
runbing.setName("aaa");
//访问成员变量
String value = (String) ReflectHelp.getFileValue(runbing, "sex");
System.out.println("value:" + value);
//修改成员变量
ReflectHelp.modifyFileValue(runbing, "sex", "bbb");
System.out.println("value:" + runbing.getName());
//检查修改后的变量值
value = (String) ReflectHelp.getFileValue(runbing, "sex");
System.out.println("value:" + value);
//调用方法
value = (String) ReflectHelp.useMethod(runbing, "add", new Class[] ...{},
new Class[] ...{});
System.out.println("value:" + value);
//使用指定构造函数
Class[] inArgs = new Class[]...{String.class};
Object[] inArgsParms = new Object[]...{"hanj"};
RunBing runBing2 = (RunBing)ReflectHelp.getObjectByConstructor("com.ccit.hj.reflect.RunBing",inArgs,inArgsParms);
value = (String) ReflectHelp.getFileValue(runBing2, "name");
System.out.println("cc -- value:" + value);
}
相关文章推荐
- java反射学习笔记(3)---使用Class来获取方法、成员变量、构造函数信息
- JAVA反射中类的构造函数,成员变量,成员方法的使用
- java反射,获取类的方法,成员变量,构造函数信息
- Java反射---Class对象,获得类的方法、成员变量和构造函数信息
- 黑马程序员——【Java反射学习】方法的反射/成员变量的反射/构造函数的反射
- java 使用反射机制来获取私有方法和成员变量
- Java学习提要——"反射"获取方法信息(成员函数、成员变量 、构造函数)
- Android(java)学习笔记109:通过反射获取成员变量和成员方法并且使用
- Java使用反射来获取成员变量泛型信息
- Java基础---基础加强---增强for循环、自动拆装箱及享元、枚举的作用、实现带有构造方法、透彻分析反射的基础_Class类、成员变量的反射、数组参数的成员方法进行反射、数组的反射应用
- Java反射(四)-Java获取成员变量与构造函数信息
- 通过Java对象来遍历成员方法,成员变量,构造函数
- Java基础---基础加强---增强for循环、自动拆装箱及享元、枚举的作用、实现带有构造方法、透彻分析反射的基础_Class类、成员变量的反射、数组参数的成员方法进行反射、数组的反射应用
- java利用反射来调用一个类的私有方法和成员变量
- Java 中,为什么在静态方法中可以调用的方法、可以使用的成员变量必须是静态的?
- 反射机制的概述和字节码对象的获取,通过反射获取构造方法并使用;通过反射获取成员变量并使用;通过反射获取私有成员变量并使用;通过反射获取成员方法并使用
- JAVA基础day05 面向对象思想 成员变量和方法 封装 构造方法和this的使用
- java中使用反射做一个工具类,来为指定类中的成员变量进行赋值操作,使用与多个类对象的成员变量的赋值。
- JAVA之通过反射遍历JavaBean成员变量和方法
- 使用java 反射,对类中成员变量赋值.将Json对像转为Java对像