Java反射机制

一、什么是反射机制 
    简单的来说,反射机制指的是程序在运行时能够获取自身的信息。在java中,只要给定类的名字,那么就可以通过反射机制来获得类的所有信息。 那么就可以通过反射机制来获得类的所有信息。 

二、哪里用到反射机制 

    有些时候,我们用过一些知识,但是并不知道它的专业术语是什么,在刚刚学jdbc时用过一行代码,Class.forName("com.mysql.jdbc.Driver")但是那时候只知道那行代码是生成驱动对象实例,并不知道它的具体含义。听了反射机制这节课后,才知道,原来这就是反射,现在很多开框架都用到反射机制,hibernate、struts都是用反射机制实现的。

三、反射机制的优点与缺点 
     为什么要用反射机制?直接创建对象不就可以了吗,这就涉及到了动态与静态的概念,静态编译:在编译时确定类型,绑定对象,即通过。动态编译:运行时确定类型,绑定对象。动态编译最大限度发挥了java的灵活性,体现了多态的应用,有以降低类之间的藕合性。一句话,反射机制的优点就是可以实现动态创建对象和编译,体现出很大的灵活性,特别是在J2EE的开发中它的灵活性就表现的十分明显。比如,一个大型的软件,不可能一次就把把它设计的很完美,当这个程序编译后,发布了,当发现需要更新某些功能时,我们不可能要用户把以前的卸载,再重新安装新的版本,假如这样的话,这个软件肯定是没有多少人用的。采用静态的话,需要把整个程序重新编译一次才可以实现功能的更新,而采用反射机制的话,它就可以不用卸载,只需要在运行时才动态的创建和编译,就可以实现该功能。它的缺点是对性能有影响。使用反射基本上是一种解释操作,我们可以告诉JVM,我们希望做什么并且它满足我们的要求。这类操作总是慢于只直接执行相同的操作。 
四、利用反射机制能获得什么信息 
      一句话,类中有什么信息,它就可以获得什么信息,不过前提是得知道类的名字,要不就没有后文了。首先得根据传入的类的全名来创建Class对象。

//第一种方法
Class forName = Class.forName("cn.zhaoqiang.reflection.UserInfo");
//第二种方法
//Class clazz =UserInfo.class;
//第三种方法
//Class clazz = new UserInfo().getClass();

//1.名和类名同时获取
    public static void getPackageAndClassName(Class clazz){
        Package pack = clazz.getPackage();
        System.out.println("获取类名与包名");
        System.out.println("类名为:"+clazz.getSimpleName());
        System.out.println("包名为:"+pack.getName());
        System.out.println("类的全名为:"+clazz.getName());
        
    }

//2.获取类中的字段
    public static void getClassFields(Class clazz) throws Exception{
    //getDeclaredFields() 只能获取当前类的字段
        Field[] declaredFields = clazz.getDeclaredFields();
        //不知道类名的情况下也可以创建类的对象
        Object obj = clazz.newInstance();
        for (Field field : declaredFields) {
            field.setAccessible(true);
            System.out.println("获取字段名称:"+field.getName()); 
            //取出字段保存的值
            System.out.println("获取字段中保存的值:"+field.get(obj));
        }
            Field field = clazz.getDeclaredField("age");
            field.setAccessible(true);
            //给字段赋值
            field.set(obj, 12);
           System.out.println("赋值成功!值为:"+ field.get(obj));
           //获取age字段的数据类型
            Class<?> type = field.getType();
           System.out.println("age字段的数据类型是:"+type.getName());
           //获取访问修饰符
           int code = field.getModifiers();
           System.out.println("该字段的访问修饰符是:"+Modifier.toString(code));
}

//3:获取类中的方法
            public static void getClassMethod(Class clazz) throws Exception{
                   //可以获取从父类继承过来的方法
                    Method[] methods = clazz.getMethods();
                    
                    System.out.println("方法的个数:"+methods.length);
                    for (Method method : methods) {
                        System.out.println("类和父类的方法有:"+method.getName());
                    }
                    System.out.println("--------------------------------------------------------");
                    //获取当前类的方法
                    Method[] declaredMethods = clazz.getDeclaredMethods();
                    for (Method method : declaredMethods) {
                           System.out.println("该类的方法有:"+method.getName());
                    }
                    //获取单个方法
                    Method method = clazz.getDeclaredMethod("say",String.class);
                    //获取方法的返回值
                    Class  clazzs = method.getReturnType();
                    System.out.println("该方法的返回值是"+clazzs.getName());
                    //获取方法的形参类型
                    Class[] params = method.getParameterTypes();
                    for (Class class1 : params) {
                        System.out.println("形参有:"+class1.getName());
                    }
                    //不知道类名的情况下也可以创建类的对象
                    Object obj = clazz.newInstance();
                    //invoke让method对象执行
                    Object obj1= method.invoke(obj, "我是狗");
                    System.out.println("方法执行后的结果是:"+obj1);
            }

//4:获取构造
            public static void getClassConstructor(Class<?> clazz) throws Exception{
                //获取所有构造器
                  Constructor[] constructors = clazz.getConstructors();
                  for (Constructor<?> constructor : constructors) {
                        System.out.println(constructor.getName());
                  }
                  //获取单个构造器
                  Constructor<?> constructor = clazz.getConstructor(String.class,int.class);
                  //执行构造器,返回类的实例
                  Object obj = constructor.newInstance("呵呵",12);
            }

 

//5:获取注解
             public static void getClassAnnotion(Class clazz) throws Exception{
                       @SuppressWarnings("unchecked")
                    Method method = clazz.getDeclaredMethod("getAge");
                       java.lang.annotation.Annotation[] annotations = method.getAnnotations();
                       for (java.lang.annotation.Annotation annotation : annotations) {
                          System.out.println("该方法的注解有:"+annotation.toString());
                    }
             }

//6:获取类的父类以及实现的接口
        public static void getSuperClassAndInterface(Class clazz){
            System.out.println("该类的父类是:"+clazz.getSuperclass().getName());
            System.out.println("该类实现的接口有:");
            Class[] interfaces = clazz.getInterfaces();
            for (Class item : interfaces) {
                System.out.println("接口名称为:"+item.getName());
            }
            
        }

 

--常见方法总结:
clazz.getPackage() 返回值是Package对象,获取包
clazz.getName() 返回值String,获取完整类名
clazz.getSimpleName() 返回值String 获取完整类名
clazz.getFields/getField(String 字段名) 返回值 Field获取类中的字段(包括父类的)
clazz.getDeclaredFields/getDeclaredField(String 字段名) 返回值 Field 仅仅获取类的字段
clazz.getSuperclass() 返回值Class 获取父类
clazz.getInterfaces() 返回Class[] 获取实现的接口
clazz.getMethods/getMethod(String 方法名,Class 形参类型) 获取方法(包括父类的方法)
clazz.getDeclaredMethods/getDeclaredMethod(String 方法名,Class 形参类型) 获取当前类的方法,不包括父类
method.invoke(Object obj) 执行方法 返回值Object
method.getReturnType() 返回值Class 获取方法的返回值类型
method.getParameterTypes() 返回值Class[] 获取方法的形参数组
field.get(Object) 获取字段的值 返回值Object
field.set(Object,value) 给字段赋值
field.getType() 返回值Class 获取字段类型
field.getModifiers() 获取字段的访问修饰符码 十六进制
Modifier.toString(int) 将code返回成真实的修饰符 private
clazz.newInstance():创建类的实例
clazz.getConstructors(): 返回值Constructor[] 获取类中的所有构造
clazz.getConstructor(Class) 获取单个构造器 返回值Constructor
Constructor.newInstance(Object) 运行构造器 返回值是类的实例
clazz.getAnnotions() Annotion[] 获取所有注解

       总的来说,java反射机制是一个很好用的东西,用它可以解决很多死的东西,因为反射机制的灵活行很大,有了他,我们就不要花太多的时间来写操做数据库的代码了,而是方法更多的时间在项目的逻辑功能上,这个可以很大的减少开发时间,而且代码的可读性好。先在的很多开源框架都是才用的反射机制,它只要配置文件,然后按规则来调用他的方法就可以了。

  
    
     
    
     
     
     
    
     
          

 
 

posted @ 2017-08-01 13:33  心与心的距离  阅读(220)  评论(0编辑  收藏  举报