Java在运行时识别类型信息的方法详解

  

  

在日常的学习工作当中,有一些知识是我们在读书的时候就能够习得;但有一些知识不是的,需要在实践的时候才能得到真知——这或许就是王阳明提倡的“知行合一”。

  

在Java中,并不是所有的类型信息都能在编译阶段明确,有一些类型信息需要在运行时才能确定,这种机制被称为RTTI,英文全称为运行时类型识别,即运行时类型识别,有没有一点“知行合一”的味道?运行时类型识别主要由类类实现。

  


  

  

在Java中,我们常用“类”(首字母为小写的c)关键字来定义一个类,说这个类是对某一类对象的抽象。你比如说王二是一个网络知名作者,我们可以这样简单地定义作者类:

        包com.cmower.java_demo.fifteen;      类作者{   私人字符串pen_name;   私人字符串real_name;   }      

现在,我们想知道作家这个类本身的一些信息(比如说类名),该怎么办呢?这时候就需要用到“类”(首字母为大写的C)类,该类包含了与类有关的信息。请看以下代码:

        公共类测试{   公共静态void main (String [] args) {   作者作者如一=new ();   类c1=wanger.getClass ();   System.out.println (c1.getName ());//输出com.cmower.java_demo.fifteen.Author   }   }      

当我们创建了作者对象如一后,就可以通过wanger.getClass()获取如一的类对象,通过c1.getName()可获得如一对象的类名。

  

想象一下,经过五年的刻意练习,王二从一名写作爱好者晋升为一名作家了。我们用代码来假装一下:

        包com.cmower.java_demo.fifteen;      类作者{   私人字符串pen_name;   私人字符串real_name;   }      类作家扩展作者{   私人字符串荣誉;   }      公共类测试{   公共静态void main (String [] args) {   作者如一=新作家();   类c1=wanger.getClass ();   System.out.println (c1.getName ());//输出com.cmower.java_demo.fifteen.Writer   }   }      

在上例中,即使我们将作家的对象引用如一向上转型为作者,如一的类对象类型依然是作家(通过输出结果可以判定)。这也就是说,Java能够在运行时自动识别类型的信息,它不会因为如一的引用类型是作者而丢失如一真正的类型信息(作家). Java是怎么做到这一点呢?

  

当Java创建某个类的对象,比如作家类对象时,Java会检查内存中是否有相应的类对象。如果内存中没有相应的类对象,那么Java会在. Class文件中寻找作家类的定义,并加载作家类的类对象。

  

一旦类对象加载成功,就可以用它来创建这种类型的所有对象。这也就是说,每个对象在运行时都会有对应的类对象,这个类对象包含了这个对象的类型信息,因此,我们能够通过类对象知道某个对象“真正”的类型,并不会因为向上转型而丢失。

  


  

  

在使用getClass()方法获取一个类的类对象时,我们必须要先获取这个类的对象,比如上面提到的如一。如果我们之前没有获取这个类的对象,就需要用另外两种方式来获取类的类对象:

        类c2=Writer.class;   System.out.println (c2.getName ());      尝试{   类c3=forname (“com.cmower.java_demo.fifteen.Writer”);   System.out.println (c3.getName ());   }捕捉(ClassNotFoundException e) {   e.printStackTrace ();   }      

1)当使用. Class来获取类对象时,不会自动地初始化该课程对象,初始化被延迟到了对静态方法或者非最终静态域进行首次引用时才执行。这样做不仅更简单,而且更安全,因为它在编译时就会受到检查(因此不需要置于尝试语句块中)。

  

2) forname会自动地初始化该课程对象,但需要指定类名,并且需要置于尝试语句块中。

  


  

  

类类为我们提供了一些非常有用的方法,比如说getName()用来返回类名,getPackage()返回类所在的包名。

  

我们还可以利用类类提供的newInstance()方法来创建相应类的对象,比如:

        类c2=Writer.class;   System.out.println (c2.getName ());      尝试{   作家wangsan=(作家)c2.newInstance ();   System.out.println (wangsan);//输出:com.cmower.java_demo.fifteen.Writer@7852e922   }捕捉(InstantiationException | IllegalAccessException e1) {   e1.printStackTrace ();   }

Java在运行时识别类型信息的方法详解