在java.lang.reflect。主要功能:在运行时
- 判断任意一个对象所属的类
- 构造任意一个类的对象
- 判断任意一个类具有的成员变量和方法
- 调用任意一个对象的方法
- 生成动态代理
反射可以访问的常用信息:
访问成员变量
BigDecimal bd = new BigDecimal("11111111111111234");
Class c = bd.getClass(); // 先getClass()之后再访问这些函数
Field[] fi = c.getFields();
// 其中Field对象常用的方法如下面一段程序所示:(用Field对象访问类成员变量)
public void contextLoads() {
DBUtil dbu = new DBUtil();
Class c1 = dbu.getClass();
// get all variables
Field[] declaredFields = c1.getDeclaredFields();
for (int i = 0; i < declaredFields.length; i++) {
Field f = declaredFields[i];
// get variable's name
System.out.println("Name: " + f.getName());
Class fieldType = f.getType();
// get its Class
System.out.println("Type: " + fieldType);
boolean isTurn = true;
while(isTurn) {
try {
isTurn = false;
// get(): get vals
System.out.println("Before modified: " + f.get(dbu));
if(fieldType.equals(int.class)) {
// static can be modified
System.out.println("use setInt()");
f.setInt(dbu, 100);
// getInt(Object obj)
} else if(fieldType.equals(float.class)) {
System.out.println("use setFloat()");
f.setFloat(dbu, 29.815f);
// getFloat(Object obj)
} else if(fieldType.equals(boolean.class)) {
System.out.println("use setBoolean()");
f.setBoolean(dbu, true);
// getBoolean(Object obj)
} else {
// other types
System.out.println("use set()");
f.set(dbu, "test1009");
}
System.out.println("After modified: " + f.get(dbu));
} catch(Exception e) {
System.out.println("Exception. Use setAccessible()");
// then private and protected can also be accessible
f.setAccessible(true);
isTurn = true;
}
}
System.out.println("========end=======\n");
}
}
访问构造函数
// 先自定义一个类 Book
public class Book {
String name;
int id, price;
// default constructor
private Book() {}
// 可变参数
public Book(String...strings)throws NumberFormatException {
if (0 < strings.length) {
id = Integer.valueOf(strings[0]);
}
if (1 < strings.length) {
price = Integer.valueOf(strings[1]);
}
}
// 俩参数
protected Book(String _name, int _id) {
this.name = _name;
this.id = _id;
}
public void print() {
System.out.println("name: " + name);
System.out.println("id: " + id);
System.out.println("price: " + price);
}
}
// 然后写反射访问所有构造函数
public void contextLoads() {
Class book = Book.class;
Constructor[] declaredConstructors = book.getDeclaredConstructors();
for (int i = 0; i < declaredConstructors.length; i++) {
Constructor con = declaredConstructors[i];
// 可变参数吗
System.out.println("查看是否允许带可变数量的参数:"+con.isVarArgs());
// con.getModifiers()返回int,可以用isStatic(int) isPublic(int) isProtected(int)
// isPrivate(int) isFinal(int)去判断,或者直接转成String输出
System.out.println("修饰符:" + Modifier.toString(con.getModifiers()));
System.out.println("该构造方法的入口参数类型依次为:");
//获取所有参数类型
Class[] parameterTypes=con.getParameterTypes();
for (int j = 0; j < parameterTypes.length; j++) {
System.out.print(" " + parameterTypes[j] + " ");
}
System.out.println("\n该构造方法可能拋出的异常类型为:");
//获取所有可能拋出的异常类型
Class[] exceptionTypes=con.getExceptionTypes();
for(int j = 0; j < exceptionTypes.length; j++) {
System.out.print(" " + parameterTypes[j] + " ");
}
Book book1 = null;
while (book1 == null) {
try {
// 访问权限是private时,抛异常
if (i == 0) {
// 顺序是类实现里的倒序
book1 = (Book)con.newInstance("Java Tutorial", 100);
= }= else if (i == 2) {
book1 = (Book)con.newInstance();
} else {
// 这里params要跟类定义里一致,不然一直无法实例化,就会死循环
Object[] params = new Object[] {new String[] {"100", "200"}};
book1 = (Book)con.newInstance(params);
}
} catch (Exception e) {
System.out.println("Access denied. Use setAccessible()");
// 授予权限
con.setAccessible(true);
}
}
book1.print();
System.out.println("\n======================\n");
}
}
访问其他方法
类似访问构造函数,常用方法有:
// 实例:为上面的 Book 类添加几个方法:
//static 作用域方法
static void staticMethod()
{
System.out.println("执行staticMethod()方法");
}
//public 作用域方法,带一个参数
public int publicMethod(int i)
{
System.out.println("执行publicMethod()方法");
return 100+i;
}
//protected 作用域方法,带两个参数
protected int protectedMethod(String s,int ==i)throw=s NumberFormatException
{
System.out.println("执行protectedMethod()方法");
return Integer.valueOf(s)+i;
}
//private 作用域方法,带可变参数
private String privateMethod(String ...strings)
{
System.out.println("执行privateMethod()方法");
// StringBuffer是可以修改的,其实可以用StringBuilder
StringBuffer sb=new StringBuffer();
for(int i=0;i<strings.length;i++)
{
sb.append(strings[i]);
}
return sb.toString();
}
// ======================================================================
// 用反射访问方法
public void contextLoads() {
Book book = new Book();
Class class1 = book.getClass();
//get all methos
// java.lang.reflect
Method[] declaredMethods = class1.getDeclaredMethods();
for(int i = 0; i < declaredMethods.length; i++) {
Method method = declaredMethods[i];
System.out.println("方法名称为:" + method.getName());
System.out.println("方法是否带有可变数量的参数:" + method.isVarArgs());
System.out.println("方法的参数类======型依次为:");
//获取所有参数类型
Class[] methodType = method.getParameterTypes();
for(int j = 0; j < methodType.length; j++) {
System.out.println(" "+methodType[j]);
}
//获取返回值类型
System.out.println("方法的返回值类型为:" + method.getReturnType());
System.out.println("方法可能抛出的异常类型有:");
//获取所有可能抛出的异常
Class[] methodExceptions = method.getExceptionTypes();
for(int j = 0; j < methodExceptions.length; j++) {
System.out.println(" " + methodExceptions[j]);
}
boolean isTurn=true;
while(isTurn) {
try { //如果该成员变量的访问权限为private,则抛出异常
isTurn = false;
if(method.getName().equals("staticMethod")){
method.invoke(book); // 这个函数不需要参数
}
else if(method.getName().equals("publicMethod")) {
//调用一个参数的方法
System.out.println("publicMethod(10)的返回值为:" + method.invoke(book, 10));
}
else if(method.getName().equals("protectedMethod")) {
//调用两个参数的方法
System.out.println("protectedMethod(\"10\",15)的返回值为:"+method.invoke(book,"10",15));
}
else if(method.getName().equals("privateMethod")) {
//调用可变数量参数的方法
Object[] parameters=new Object[]{new String[]{"J","A","V","A"}};
System.out.println("privateMethod()的返回值为:"+method.invoke(book, parameters));
}
}
catch (Exception e) {
System.out.println("在设置成员变量值时抛出异常,下面执行setAccessible()方法");
method.setAccessible(true); //设置为允许访问private方法
isTurn = true;
}
}
System.out.println("=============================\n");
}
}
// 输出:不会访问Constructor,因为它们是单独的
方法名称为:print
方法是否带有可变数量的参数:false
方法的参数类型依次为:
方法的返回值类型为:void
方法可能抛出的异常类型有:
=============================
方法名称为:protectedMethod
方法是否带有可变数量的参数:false
方法的参数类型依次为:
class java.lang.String
int
方法的返回值类型为:int
方法可能抛出的异常类型有:
class java.lang.NumberFormatException
执行protectedMethod()方法
protectedMethod("10",15)的返回值为:25
=============================
方法名称为:staticMethod
方法是否带有可变数量的参数:false
方法的参数类型依次为:
方法的返回值类型为:void
方法可能抛出的异常类型有:
执行staticMethod()方法
=============================
方法名称为:publicMethod
方法是否带有可变数量的参数:false
方法的参数类型依次为:
int
方法的返回值类型为:int
方法可能抛出的异常类型有:
执行publicMethod()方法
publicMethod(10)的返回值为:110
=============================
方法名称为:privateMethod
方法是否带有可变数量的参数:true
方法的参数类型依次为:
class [Ljava.lang.String;
方法的返回值类型为:class java.lang.String
方法可能抛出的异常类型有:
在设置成员变量值时抛出异常,下面执行setAccessible()方法
执行privateMethod()方法
privateMethod()的返回值为:JAVA
远程调用方法需要借助对象和socket