Java反射

  这几天又重新学习了一下反射,略有感悟(以下是学习笔记):

镜像:类被加载到内存以后,通过类镜像来表示,java.lang.Class
反射:在运行时,通过类镜像进行类操作(获取类信息、构建类对象、调用方法等),使用java.lang.reflect包下的API(Field、Method、Constructor)

反射使用场景:当类的信息在编译时无法确定,运行时才能确定下来时,必须使用反射进行相关操作

反射编程步骤:
1)获取类镜像:java.lang.Class.forName(...)
2)如需要,创建类实例对象:Class-->newInstance()
3)调用反射相关API:
java.lang.reflect.Field
java.lang.reflect.Constructor
java.lang.reflect.Method

  以下是一个当你不知道客户所用数据库类型时,依靠反射来获取类镜像的过程:

//这是db.cfg文件,要从中获取类名与方法名
driver=com.wsy.refluction.OracleDriver
method=load

//以下是Java程序
package com.wsy.refluction;

import java.io.FileInputStream;
import java.lang.reflect.Method;
import java.util.Properties;

interface Driver{
    public void load(String info , String name);
}

class OracleDriver implements Driver{

    @Override
    public void load(String info , String name) {
        System.out.println("This is OracleDriver " + "info:" + info + " " + "name:" + name);
    }
    
}

class MySQLDriver implements Driver{

    @Override
    public void load(String info , String name) {
        System.out.println("This is MySQLDriver");
    }
    
}

class SQLServerDriver implements Driver{

    @Override
    public void load(String info , String name) {
        System.out.println("This is SQLServerDriver");
    }
    
}
public class Refluction {
    public static void main(String[] args) {
        Properties prop = new Properties();
        try {
            prop.load(new FileInputStream("src/com/wsy/refluction/db.cfg"));
            String driverName = prop.getProperty("driver");
            String methodName = prop.getProperty("method");
            //结果:com.wsy.refluction.OracleDriver
            System.out.println(driverName);
            Class<?> c = Class.forName(driverName);
            Object o = c.newInstance();
            //结果:com.wsy.refluction.OracleDriver@15db9742
            System.out.println(o);
            Method[] methods = c.getMethods();
            for(Method method : methods) {
                if(method.getName().equals(methodName)) {
                    //结果:This is OracleDriver
                    //调用成员方法
                    method.invoke(o , new Object[] {"driver" , c.getName()});
                    //调用静态方法
                    //method.invoke(null , new Object[] {"driver" , c.getName()});
                    
                }
            }
            
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

  下面这个程序是通过反射来获取一个类的各种信息:

//随意定义一个类(这里以Student类为例)
package com.wsy.refluction;

public class Student implements Comparable<Object>{
    private String name;
    private int age;
    public Student(){}
    public Student(String name,int age){
        this.name = name;
        this.age = age;
    }
    public String getName(){
        return name;
    }
    //public void setName(String name){
    private void setName(String name){
        this.name = name;
    }
    public int getAge(){
        return age;
    }
    public void setAge(){
        this.age = age;
    }
    public String toString(){
        StringBuffer sb = new StringBuffer();
        sb.append("name:");
        sb.append(name);
        sb.append(",age:");
        sb.append(age);
        return sb.toString();
    }
    public boolean equals(Object o){
        if(!(o instanceof Student))
            return false;
        Student s = (Student)o;
        return ((s.name.equals(name)) && (s.age == age));
    }
    public int hashCode(){
        return name.hashCode() + age;
    }
    public int compareTo(Object o){
        Student s = (Student)o;
        return s.age - age;
    }
    public static void talk(){
        System.out.println("in Student talk()");
    }
}

//以下是主程序
package com.wsy.refluction;

import java.lang.reflect.*;

public class ReflectionTest2{
    public static void printClass(Class clazz){
        Package pac = clazz.getPackage();
        String pacName = pac.getName();
        System.out.println("package " + pacName + ";");
        int modify = clazz.getModifiers();
        String modifier = Modifier.toString(modify);
        String className = clazz.getName();
        boolean isInterface = clazz.isInterface();
        String classFlag = "";
        if(isInterface)
            classFlag = "interface";
        else
            classFlag = "class";
        System.out.println(modifier + " " + classFlag + " " + className + "{");
    }
    public static void printField(Class clazz){
        Field[] fields = clazz.getDeclaredFields();
        String modifier;
        String type;
        String fieldName;
        for(int i = 0;i < fields.length;i++){
            modifier = Modifier.toString(fields[i].getModifiers());    
            type = fields[i].getType().getName();
            fieldName = fields[i].getName();
            System.out.println("\t" + modifier + " " + type + " " + fieldName + ";");
        }
    }
    public static void printConstructor(Class clazz){
        Constructor[] cons = clazz.getDeclaredConstructors();
        String consInfo;
        for(int i = 0;i < cons.length;i++){
            consInfo = cons[i].toGenericString();
            System.out.println("\t" + consInfo + "{...}");
        }
    }
    public static void printMethod(Class clazz){
        Method[] methods = clazz.getDeclaredMethods();
        String methodInfo;
        for(int i = 0;i < methods.length;i++){
            methodInfo = methods[i].toGenericString();
            System.out.println("\t" + methodInfo + "{...}");
        }
        System.out.println("}");
    }
    public static void main(String[] args)throws Exception{
        if(args.length != 1){
            System.out.println("Please input argument!");
            System.exit(1);
        }
        Class clazz1 = Class.forName(args[0]);
        ClassLoader loader = new ClassLoader(){};
        Class clazz2 = loader.loadClass(args[0]);
        System.out.println("clazz1:" + clazz1);
        System.out.println("clazz2:" + clazz2);
        Student stu = new Student();
        Class clazz3 = stu.getClass();
        Class clazz4 = com.wsy.refluction.Student.class;
        System.out.println("clazz3:" + clazz3);
        System.out.println("clazz4:" + clazz4);
        
        System.out.println("*********************");
        printClass(clazz1);
        printField(clazz1);
        printConstructor(clazz1);
        printMethod(clazz1);

        System.out.println("*********************");
        Constructor[] cons = clazz1.getConstructors();
        Constructor con = null;;
        for(int i = 0;i < cons.length;i++){
            if(cons[i].getParameterTypes().length != 2)
                continue;
            con = cons[i];
        }
        Student s = (Student)con.newInstance(new Object[]{"Jack",20});
        System.out.println("studnet:" + s);

        Method[] methods = clazz1.getDeclaredMethods();
        Method method = null;
        Method method2 = null;
        for(int i = 0;i < methods.length;i++){
            if("setName".equals(methods[i].getName())){
                method = methods[i];
            }
            if("talk".equals(methods[i].getName())){
                method2 = methods[i];
            }
        }
        // 如果是静态方法,第一个参数传null,如果没有参数,第二个传null
        method.setAccessible(true);
        method.invoke(s, new Object[]{"Tom"});
        System.out.println(s);
        method2.invoke(null,new Object[]{});    
    }
}


//以下是运行结果
clazz1:class com.wsy.refluction.Student
clazz2:class com.wsy.refluction.Student
clazz3:class com.wsy.refluction.Student
clazz4:class com.wsy.refluction.Student
*********************
package com.wsy.refluction;
public class com.wsy.refluction.Student{
    private java.lang.String name;
    private int age;
    public com.wsy.refluction.Student(){...}
    public com.wsy.refluction.Student(java.lang.String,int){...}
    public boolean com.wsy.refluction.Student.equals(java.lang.Object){...}
    public java.lang.String com.wsy.refluction.Student.toString(){...}
    public int com.wsy.refluction.Student.hashCode(){...}
    public int com.wsy.refluction.Student.compareTo(java.lang.Object){...}
    public java.lang.String com.wsy.refluction.Student.getName(){...}
    private void com.wsy.refluction.Student.setName(java.lang.String){...}
    public static void com.wsy.refluction.Student.talk(){...}
    public void com.wsy.refluction.Student.setAge(){...}
    public int com.wsy.refluction.Student.getAge(){...}
}
*********************
studnet:name:Jack,age:20
name:Tom,age:20
in Student talk()

 

posted @ 2017-09-03 15:42  渊源谭  阅读(204)  评论(0编辑  收藏  举报