Java反射机制浅析
概述:
  JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法和属性;这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制。
本文链接:http://blog.csdn.net/lemon_tree12138/article/details/49871515
 -- 
Coding-Naga
                                                                 --转载请注明出处
反射的基本操作:
1.获得类名
通过Java的反射机制我们可以获得一个对象的类。这里的类是类的全名,包含包名。比如:org.demo.reflect.Clazz.
可能你会问,这个功能有什么用?其实这个功能还是很有用的。比如,我们在打日志的时候,我们就要用到这个功能,来追踪这个日志是在什么地方被打出来的。
private String getObjectName(Object o) {
        if (o == null) {
            return null;
        }
        
        return o.getClass().getName();
    }
2.输出一个类的所有对外可见的变量
  注意这里是输出所有对外可见的变量,即变量的修饰符为public.这里使用的是getFields()方法。注意,这里是可以获得其父类中的public变量.
private void printFields(String clazzName) {
        try {
            Class<?> clazz = Class.forName(clazzName);
            Field[] fields = clazz.getFields();
            for (Field field : fields) {
                System.out.println(field);
            }
        } catch(ClassNotFoundException e) {
            e.printStackTrace();
        }
    }
3.输出一个类的所有成员变量
这里是输出一个类的成员变量,不论它是不是对外可见的。就算它是被private修饰,也同样可见。这一点很让人着迷,不是吗?
  这里使用的是getDeclaredFields()。
private void printDeclaredFields(String clazzName) {
        try {
            Class<?> clazz = Class.forName(clazzName);
            Field[] fields = clazz.getDeclaredFields();
            for (Field field : fields) {
                System.out.println(field);
            }
        } catch(ClassNotFoundException e) {
            e.printStackTrace();
        }
    }
4.输出一个类的所有public方法
  这里与之前的getFields()类似只能对外可见的,即修饰符为public的方法。同样,这里也是可以获得父类的public方法。
private void printMethods(String clazzName) {
        try {
            Class<?> clazz = Class.forName(clazzName);
            Method[] methods = clazz.getMethods();
            for (Method method : methods) {
                System.out.println(method);
            }
        } catch(ClassNotFoundException e) {
            e.printStackTrace();
        }
    }
5.输出一个类的所有成员方法
private void printDeclaredMethods(String clazzName) {
        try {
            Class<?> clazz = Class.forName(clazzName);
            Method[] methods = clazz.getDeclaredMethods();
            for (Method method : methods) {
                System.out.println(method);
            }
        } catch(ClassNotFoundException e) {
            e.printStackTrace();
        }
    }
6.调用类里面的成员函数
这里我先假设测试类中有一个sayHello方法,如下:
public void sayHello(String from) {
    System.out.println("Hello from " + from);
}
下面就可以通过getMethod()方法来反射获得这个sayHello方法。getMethod()中第一个参数为方法名,这里即是"sayHello",后面的参数是可变量参数,是可以填写0个或是多个的,值为被反射的方法中的参数类型。这里即是String.class。
  然后再调用invoke()方法,invoke()的第一个参数为被反射的对象,后面是向sayHello()方法中传递的数据。具体过程如下:
private void callMethod(Object o, String clazzName) {
        try {
            Class<?> clazz = Class.forName(clazzName);
            Method method = clazz.getMethod("sayHello", String.class);
            method.invoke(o, TestReflect.class.getSimpleName());
        } catch(Exception e) {
            e.printStackTrace();
        }
    }
7.生成一个不能new的对象
可以看到上面的操作过程,都有一个前提,就是我们已经有了这个对象。那如果我们之前没有这个对象呢?或者说,我们不能显式地获得这个对象。这一点在学习单例模式的时候,就知道其实很好获得,只要把重写默认的构造方法,并把这个构造方法的修饰符修改为private即可。如下:
private PrivateClass() {
}
  这里有一点比较特殊,就是这里使用的类或是方法都是私有的。我们不能直接去使用他们,在使用他们之前,需要设置其可访问性为true。具体操作过程如下:
private void newPrivateClass(String clazzName) {
        try {
            Class<?> clazz = Class.forName(clazzName);
            Constructor<?> constructor = clazz.getDeclaredConstructor();
            constructor.setAccessible(true);
            PrivateClass privateClass = (PrivateClass)constructor.newInstance();
            
            Method method = clazz.getDeclaredMethod("sayHello");
            method.setAccessible(true);
            method.invoke(privateClass);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
 
                     
                    
                 
                    
                 
                
            
         
 
         浙公网安备 33010602011771号
浙公网安备 33010602011771号