java类的反射
反射库提供了一个非常丰富且精心设计的工具集,以便编写能够动态操纵java代码的程序。
在java2核心技术中讲到:"使用他的主要对象是工具构造者,而不是应用程序员",其实对于我们应用程序员来说,偶尔利用一下java反射可以帮助我们化简我们的代码
第一种情况,我们的系统里面有很多类,你想为每个类编写toString()方法来,查看对象的所有属性。常规方法在每个类里覆盖toString()函数,去把类中的各个属性输出出来。
如果你做一个很大等程序,有很多个model类,你要把每个类添加上toString()方法,如果每个model类里面又有很多成员变量。那么要完成这一目标可能需要我们很多的时间的。
那么现在我们可以使用java提供的反射库,来减少我们的工作量。
import java.lang.reflect.*;
import java.util.*;
/**
* This program uses reflection to spy on objects.
* @version 1.11 2004-02-21
* @author Cay Horstmann
*/
public class ObjectAnalyzerTest
{
public static void main(String[] args)
{
ArrayList<Integer> squares = new ArrayList<Integer>();
for (int i = 1; i <= 5; i++)
squares.add(i * i);
System.out.println(new ObjectAnalyzer().toString(squares));
}
}
class ObjectAnalyzer
{
/**
* Converts an object to a string representation that lists all fields.
* @param obj an object
* @return a string with the object's class name and all field names and
* values
*/
public String toString(Object obj)
{
if (obj == null) return "null";
if (visited.contains(obj)) return "...";
visited.add(obj);
Class cl = obj.getClass();
if (cl == String.class) return (String) obj;
if (cl.isArray())
{
String r = cl.getComponentType() + "[]{";
for (int i = 0; i < Array.getLength(obj); i++)
{
if (i > 0) r += ",";
Object val = Array.get(obj, i);
if (cl.getComponentType().isPrimitive()) r += val;
else r += toString(val);
}
return r + "}";
}
String r = cl.getName();
// inspect the fields of this class and all superclasses
do
{
r += "[";
Field[] fields = cl.getDeclaredFields();
AccessibleObject.setAccessible(fields, true);
// get the names and values of all fields
for (Field f : fields)
{
if (!Modifier.isStatic(f.getModifiers()))
{
if (!r.endsWith("[")) r += ",";
r += f.getName() + "=";
try
{
Class t = f.getType();
Object val = f.get(obj);
if (t.isPrimitive()) r += val;
else r += toString(val);
}
catch (Exception e)
{
e.printStackTrace();
}
}
}
r += "]";
cl = cl.getSuperclass();
}
while (cl != null);
return r;
}
private ArrayList<Object> visited = new ArrayList<Object>();
}
这个类是在书里面找到的,考虑的东西比较多,看着比较复杂,大家可以根据自己的需求做些调整。当然有的同学可能不喜欢这种调用方式我们更喜欢myObject.toString();
那们我们可以设置一个父类,实现这个toString(),然后让其子类来把这个方法继承过去就可以了。
第二种情况:如果你遇到这种情况,根据一个变量的值的不同来调用不用的函数,我们以往的做法是这样子的
if("method1".equals(methodName))
Test.method1();
else if("method2".equals(methodName))
Test.method2();
else if("method3".equals(methodName))
Test.method3();
else if("method4".equals(methodName))
Test.method4();
else if("method5".equals(methodName))
Test.method5();
else if("method6".equals(methodName))
Test.method6();
else if("method7".equals(methodName))
Test.method7();
else if("method8".equals(methodName))
Test.method8();
else Test.method1();
但是这种代码看着很不专业,因此我们使用了类的反射,只需要使用三行代码
Class c=Class.forName("Test");
Method m=c.getMethod(methodName, null);
m.invoke(null, null);
当然代码可以化简是有些前提的,就是你所调用的函数是类似的,就是参数相同。否则你会发现得不偿失。