zoukankan      html  css  js  c++  java
  • Java基础之反射

    • Class类的使用
    • 动态加载类
    • 方法信息的反射
    • 获取成员变量&构造函数
    • 方法反射的基本操作
    • 通过反射了解集合泛型的本质

    一、Class类的使用

    Class 类:

    在面向对象的世界里,万事万物皆为对象

    Java语言中,静态成员、普通数据类型是不是对象呢?

    普通数据类型不是对象但是他们有他们的包装类;而静态成员是属于类的而不属于对象。

    那么,类是谁的对象呢?

    类是对象,类是java.lang.Class类的实例对象。

    Class类的实例对象,该如何表示?

    任何一个类都是Class的实例对象,这个 实例对象有3种表达方式:

    Class c1 = Demo.class
        //由此可见每个类都有隐含的静态成员属性“class”来表示该类的类类型(官方声明:表示该类的对象我们成为该类的“类 类型” class type);
    
    Demo demo = new Demo();
    Class c2 = demo.getClass();
    	//通过类对象的get方法来获取该类的类类型;
    
    Class c3 = null;
    try {
        c3 = Class.forName("io.github.newmeanning.reflect.Demo");
    } catch (ClassNotFoundException e) {
        e.printStackTrace();
    }
    

    以上三种皆表示同一个对象:

    System.out.println(c1==c2);//true
    System.out.println(c1==c3);//true
    

    如何通过类类型创建该类的实例?

    try {
        //通过表示该类 类类型.newInstance()方法来来创建
        Demo demo1 = (Demo) c1.newInstance();
    } catch (InstantiationException e) {
        e.printStackTrace();
    } catch (IllegalAccessException e) {
        e.printStackTrace();
    }
    

    总结

    万事万物皆对象;类也是一个对象,是Class类的实例对象。这个对象我们成为该类的类类型。

    二、 动态加载类

    Class.forName("类的全称")
    
    • 该方式不仅表示了类的类类型,还代表了动态加载类
    • 我们要区分编译、运行
    • 编译时刻加载类是静态加载类、运行时刻加载类是动态加载类

    而通过new创建对象是静态加载类。

    举个例子:

    public class Office {
        public static void main(String[] args) {
            if("word".equals(args[0])){
                Word w = new Word();
                w.start();
            }
            if("excel".equals(args[0])){
                Excel e = new Excel();
                e.start();
            }
        }
    }
    
    public class Word {
        public static void start(){
            System.out.println("word start");
        }
    }
    
    

    当我们运行Office的main方法的时候,提示如下信息:

    Information:java: Errors occurred while compiling module 'testSE'
    Information:javac 1.8.0_144 was used to compile java sources
    Information:2018/9/27 23:44 - Compilation completed with 2 errors and 0 warnings in 2 s 321 ms
    D:IdeaProjects	estSEsrcOffice.java
    Error:(10, 13) java: 找不到符号
      符号:   类 Excel
      位置: 类 io.github.newmeanning.reflect.Office
    Error:(10, 27) java: 找不到符号
      符号:   类 Excel
      位置: 类 io.github.newmeanning.reflect.Office
    

    我们知道Word类是存在的,而Excel类是不存在的,所以编译不通过。即便Word存在我们也无法使用。

    那如果,我们想要有什么用什么需要编译通过怎么办?

    通过动态加载类可以解决该问题:

    public class Office {
        public static void main(String[] args) {
            try {
                //动态加载类
                Class c = Class.forName(args[0]); 
                //word or excel 的统一接口
                OfficeAble officeAble = (OfficeAble) c.newInstance();
                officeAble.start();
            } catch (ClassNotFoundException e) {
                e.printStackTrace();
            } catch (IllegalAccessException e) {
                e.printStackTrace();
            } catch (InstantiationException e) {
                e.printStackTrace();
            }
        }
    }
    
    public interface OfficeAble {
        public void start();
    }
    
    
    public class Word implements OfficeAble {
        @Override
        public void start(){
            System.out.println("word start");
        }
    }
    
    

    结果如下:

    D:IdeaProjects	estSEsrc>javac *java
    
    D:IdeaProjects	estSEsrc>java Office Word
    word start
    
    D:IdeaProjects	estSEsrc>java Office Excel
    java.lang.ClassNotFoundException: Excel
            at java.net.URLClassLoader.findClass(Unknown Source)
            at java.lang.ClassLoader.loadClass(Unknown Source)
            at sun.misc.Launcher$AppClassLoader.loadClass(Unknown Source)
            at java.lang.ClassLoader.loadClass(Unknown Source)
            at java.lang.Class.forName0(Native Method)
            at java.lang.Class.forName(Unknown Source)
            at Office.main(Office.java:4)
    
    D:IdeaProjects	estSEsrc>
    
    
    

    总结&分析:

    通过new 创建对象是静态加载类,在编译时刻就需要加载所有的可能使用到的类。

    通过Class a=Class.forName(arg[0]);此时为动态加载,因为编译时不知道使用哪个类,因此编译没有加载任何类,通过编译。运行时,根据 Javac office.java word (word为arg[0],也是类类型),去确定a是哪个类。这就是动态加载。如果word不存在,此时运行会报错。

    因此我们以后编写工具类可以采用动态加载的方式。增添Excel或者PPT的时候我们只需要实现OfficeAble接口即可,不需要修改工具类的任何代码也不需要重新编译。

    三、方法信息的反射

    class.getName() 获取类名

    class.getSimpleName() 不包含包名的类的名称

    class.getMethods() 获取所有public的函数,包括由父类

    class.getDeclareMethods() 获取该类自行声明的所有方法,不论访问权限

    Method.getName() 获取方法名

    Method.getReturnType() 得到方法返回值的类型的类类型

    method.getParameterTypes():获取方法的参数类型的类类型数组class[]

    一、基本的数据类型,void关键字等都存在类类型

    ​ Class c = 基类.class (int,String,double,void等)

    二、Class类的基本API操作

    1、c.getName()可以获取类的名称
    2、c.getSimpleName();//不包含包名的类的名称
    3、c.getMethods()获取类的【public方法】集合,【包括继承来的】
    4、c.getDeclaredMethods()获取的是所有该类【自己声明】的方法,【不问访问权限】
    注意【所有方法都是Method类的对象】
    

    三、Method类提供了操作方法的方法

    1、m.getReturnType()得到该方法的返回值类型的类类型(class),如int.class String.class
    2、m.getName()得到方法的名称
    3、m.getParameterTypes()获得参数列表类型的类类型,如参数为(int,int)则得到(int.class ,int class)
    Class c1 = int.class; int的类类型
    Class c2 = String.class; String类的类类型 String类字节码
    Class c3 = double.class; double这个数据类类型的字节码表示方式
    Class c4 = Double.class; Double这个类的类类型字节码表示方式
    Class c5 = void.class; 表达了void这个类的类类型
    getName为这个类的类类型的具体名称 
    c1.getName ---> int
    c2.getName ---> java.lang.String 类的全称
    c2.getSimpleName ---> String 不包含包名的类的名称
    
    只要在类里面声明的都有类类型
    public static void pringClassMessage(Object object){
     	//要获取类的信息,首先要获取类的类型
        Class c=object.getClass();//传递的是哪个子类的对象,c就是该子类的类类型
        //获取类的名称
    
        System.out.println("类的名称是:"+c.getName());
         /*
          * Method类,方法对象
          * 一个成员方法就是一个Method对象
          * getMethods()方法获取的是所有public函数,包括父类继承而来的
          * getDeclaredMethods()获取的是所有该类自己声明的方法,不问访问权限
          * */
        Method[] ms=c.getMethods();//c.getDeclaredMethods();
        for (int i = 0; i < ms.length; i++) {
             //得到方法的返回值类型的类类型
            Class returnType=ms[i].getReturnType();
            System.out.println(returnType.getName());
             //得到方法名
            System.out.println(ms[i].getName()+"(");
             //获取参数类型-->得到的是参数列表的类型的类类型
            Class[] paramType=ms[i].getParameterTypes();
             for (Class class1: paramType) {
             	System.out.println(class1.getName()+",");
             }
             	System.out.println(")");
             } 
         }
     }
    

    四、获取成员变量&构造函数

    一、成员变量是java.lang.reflect.Field的对象

    1、Field类封装了关于成员变量的操作
    2、Field[] fs = c.getFields()方法获取所有public的成员变量Field[]信息
    3、c.getDeclaredFields获取的是该类自己声明的成员变量信息
    4、field.getType()获得成员类型的类类型
    5、field.getName()获得成员的名称

    二、构造函数是java.lang.Constructor类的对象

    1、通过Class.getConstructor()获得Constructor[]所有公有构造方法信息
    2、建议getDeclaredConstructors()获取自己声明的构造方法
    3、Constructor.getName():String
    4、Constructor.getParameterTypes():Class[]
    成员变量也是对象,是java.lang.reflect.Field的对象;

    五、方法反射的基本操作

    一、获取A类中的print(int,int)方法:

    ①要获取一个方法就是获取类的信息,获取类的信息首先要获取类的类类型
    A a1=new A(); Class c= a1.getClass();
    ②获取方法 由名称和参数列表来决定,getMethod获取的是public方法,getDelcaredMethod获取自己声明的方法
    Method m =c.getMethod(methodName,paramtypes);//paramtypes可以用数组的形式 表示new Class[]{int.class,int.class},也可以直接列举类类型

    二、方法的反射操作:

    是用m对象来进行方法调用,和a1.print(10,20)调用的方法相同 m.invoke(a1,new Object[]{10,20})

    Object o=m.invoke(对象名,参数);//方法如果没有返回值返回null,如果有返回值返回具体值,参数可用数组的方式表示,也可以直接列举,没有参数就不写

    public Class A{
        public void print(){};
        public void Print(Sting a,String b){}
        public void Print(int a,int b){};
    } 
    public Class B{
        public static void main(String[] args){
            A a1 = new A();  
            Class c= a1.getclass;
            Method getMet=c.getMethod("print",String.class,String.class);//忘了加引号
            Object obj=getMet.invoke(a1,"df","df");
        }
    }
    

    六、通过反射了解集合泛型的本质

    1:反射的操作都是编译之后的操作;就是运行阶段
    2:java中集合的泛型是防止错误输入的;只在编译阶段有效,只要绕过编译就无效啦
    我们可以通过方法的反射来操作,绕过编译
    eg:

    ArrayList list1=new ArrayList();
    ArrayList<String> list2=new ArrayList<String>();
    Class c1=list1.getClass();
    Class c2=list2.getClass();
    System.out.print(c1==c2);//true
    Method m=c2.getMethod("add",Object.class);
    m.invoke(list2,20);//向list2集合中添加一个int 型的值;绕过编译
    

    当然是不能直接foreach list2集合的,会报类型转换错误

  • 相关阅读:
    Hive UDF 用户自定义函数 编程及使用
    Hive 常用命令和语句
    Hive 配置显示表头和数据库信息
    Hive 安装配置
    Hadoop完全分布式集群环境搭建
    Java 文件切割工具类
    MongoDB 安装配置
    NodeJS 安装配置
    Java 截屏工具类
    Maven 配置本地依赖jar
  • 原文地址:https://www.cnblogs.com/nm666/p/9763021.html
Copyright © 2011-2022 走看看