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

      
        
         
        
         
         
         
        
         
              

     
     

  • 相关阅读:
    onmouseover事件
    ref:ThinkPHP Builder.php SQL注入漏洞(<= 3.2.3)
    ref:mysql命令大全
    ref:mysql丢失密码,如何修改?
    ref:学习笔记 UpdateXml() MYSQL显错注入
    转:[译]CSV 注入:被人低估的巨大风险
    转:深入剖析 JavaScriptCore
    转:Exploiting Electron RCE in Exodus wallet
    转:LNMP虚拟主机PHP沙盒绕过/命令执行(php exec命令被禁之后)
    转:Ubuntu16.04下配置php+vscode+xdebug开发环境
  • 原文地址:https://www.cnblogs.com/wqtzhaoqiang/p/7268079.html
Copyright © 2011-2022 走看看