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

前言

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

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

01 Class类

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

package com.cmower.java_demo.fifteen;

class Author {
 private String pen_name;
 private String real_name;
}

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

public class Test {
 public static void main (String [] args) {
  Author wanger = new Author();
  Class c1 = wanger.getClass();
  System.out.println(c1.getName());
  //输出 com.cmower.java_demo.fifteen.Author
 }
}

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

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

package com.cmower.java_demo.fifteen;

class Author {
 private String pen_name;
 private String real_name;
}

class Writer extends Author {
 private String honour;
}

public class Test {
 public static void main (String [] args) {
  Author wanger = new Writer();
  Class c1 = wanger.getClass();
  System.out.println(c1.getName());
  //输出 com.cmower.java_demo.fifteen.Writer
 }
}

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

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

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

02 获取Class对象的其他方式

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

Class c2 = Writer.class;
System.out.println(c2.getName());

try {
 Class c3 = Class.forName("com.cmower.java_demo.fifteen.Writer");
 System.out.println(c3.getName());
} catch (ClassNotFoundException e) {
 e.printStackTrace();
}

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

2)Class.forName会自动地初始化该Class对象,但需要指定类名,并且需要置于try语句块中。

03 Class类提供的常用方法

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

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

Class c2 = Writer.class;
System.out.println(c2.getName());

try {
 Writer wangsan = (Writer) c2.newInstance();
 System.out.println(wangsan);
 // 输出:com.cmower.java_demo.fifteen.Writer@7852e922
} catch (InstantiationException | IllegalAccessException e1) {
 e1.printStackTrace();
}

由于我们在创建Class对象c2时没有使用泛型,所以newInstance()返回的对象类型需要强转为Writer。我们可以在此基础上进行改进,示例如下:

Class<Writer> c4 = Writer.class;
System.out.println(c4.getName());

try {
 Writer wangsan = c4.newInstance();
 System.out.println(wangsan);
 // 输出:com.cmower.java_demo.fifteen.Writer@7852e922
} catch (InstantiationException | IllegalAccessException e1) {
 e1.printStackTrace();
}

04 反射

我们还可以通过getFields()获取所有public修饰的字段,通过getMethods()返回所有public修饰的方法。

甚至,我们还可以通过getDeclaredFields()获取更多字段,包括公共、受保护、默认(包)访问和私有字段,但不包括继承字段。对应的,getDeclaredMethods()用来获取更多方法。示例如下:

package com.cmower.java_demo.fifteen;

import java.lang.reflect.Field;
import java.lang.reflect.Method;

class Author {
 private String pen_name;
 private String real_name;
}

class Writer extends Author {
 private String honour;

 private void makeMoney() {
  System.out.println("很多很多钱");
 }
}

public class Test {
 public static void main(String[] args) {

  Class<Writer> c4 = Writer.class;
  System.out.println(c4.getName());

  try {
   Writer wangsan = c4.newInstance();
   System.out.println(wangsan);

   Field[] fields = c4.getDeclaredFields();
   for (Field field : fields) {
    System.out.println(field.getName());
   }

   Method[] methods = c4.getDeclaredMethods();
   for (Method method : methods) {
    System.out.println(method.getName());
   }
  } catch (InstantiationException | IllegalAccessException e1) {
   e1.printStackTrace();
  }

 }
}

上面的例子其实涉及到了反射,Field、Method(还有例子中未提到的Constructor)都来自java.lang.reflect类库。Class类与java.lang.reflect类库一起对反射的概念进行了支持。

有时候,我们需要从磁盘文件或网络文件中读取一串字节码,并把它转换成一个类,这时候就需要用到反射。最常见的典型例子就是将一串JSON字符串(在网络传输中最初的形态可能是字节数组)反射为对应类型的对象。

阿里巴巴提供的FastJSON提供了 toJSONString() 和 parseObject() 方法来将 Java 对象与 JSON 相互转换。调用toJSONString方法即可将对象转换成 JSON 字符串,parseObject 方法则反过来将 JSON 字符串转换成对象。FastJSON的内部其实用的就是反射机制。

package com.cmower.common.util;

import java.io.UnsupportedEncodingException;

import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;

import com.alibaba.fastjson.JSON;

@SuppressWarnings("all")
public class JsonUtil {
 private static Log logger = LogFactory.getLog("json");

 public static byte[] objectToByte(Object obj) throws UnsupportedEncodingException {
  String jsonStr = JSON.toJSONString(obj);
  logger.debug("序列化后数据:" + jsonStr);
  return jsonStr.getBytes("UTF-8");
 }

 public static <T> T byteToObject(byte[] data, Class<T> obj) throws UnsupportedEncodingException {
  String objectString = new String(data, "UTF-8");
  logger.debug("反序列化后数据 : " + objectString);
  return JSON.parseObject(objectString, obj);
 }

 public static <T> Object stringToObject(String data, Class<T> obj) throws UnsupportedEncodingException {
  logger.debug("反序列化后数据 : " + data);
  return JSON.parseObject(data, obj);
 }
}

05 总结

为了完成这篇文章,我特意和青苗谷的一名技术专家聊了聊,问他了几个很傻逼的问题:“‘运行时'是什么意思?是站在Java虚拟机的角度,还是程序员的角度?”

他给了我很好的解释和启发,我不由觉得非常的惭愧,作为一名年纪颇长的Java学习者,竟然对理论知识薄弱到令人发指的地步——不知道你是否也有这样的困惑?

但写作的好处就在于此,在向读者解释“Java如何在运行时识别类型信息”的过程中,我的思路逐渐地清晰了起来——这真是一个自我提升的好办法!

好了,以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对我们的支持。

(0)

相关推荐

  • Java中的接口回调实例

    定义: /** * @author Administrator * @project: TestOne * @package: PACKAGE_NAME * @date: 2018/11/30 0030 15:42 * @brief: 郭宝 **/ public class Person { /** * 自定义一个接口 **/ public interface OnNameChangeListener{ //接口中的抽象函数,并携带数据 void onNameChange(String name

  • Java并发的CAS原理与ABA问题的讲解

    CAS原理 在计算机科学中,比较和交换(Compare And Swap)是用于实现多线程同步的原子指令. 它将内存位置的内容与给定值进行比较,只有在相同的情况下,将该内存位置的内容修改为新的给定值. 这是作为单个原子操作完成的. 原子性保证新值基于最新信息计算; 如果该值在同一时间被另一个线程更新,则写入将失败. 操作结果必须说明是否进行替换; 这可以通过一个简单的布尔响应(这个变体通常称为比较和设置),或通过返回从内存位置读取的值来完成(摘自维基本科) CAS流程 以AtomicIntege

  • Java内存区域和内存模型讲解

    一.Java内存区域 方法区(公有):用户存储已被虚拟机加载的类信息,常量,静态常量,即时编译器编译后的代码等数据.异常状态 OutOfMemoryError. 堆(公有):是JVM所管理的内存中最大的一块.唯一目的就是存放实例对象,几乎所有的对象实例都在这里分配.Java堆是垃圾收集器管理的主要区域,因此很多时候也被称为"GC堆".异常状态 OutOfMemoryError. 虚拟机栈(线程私有): 描述的是java方法执行的内存模型:每个方法在执行时都会创建一个栈帧,用户存储局部变

  • 小米推送Java代码

    maven <dependency> <groupId>com.xiaomi</groupId> <artifactId>json-simple</artifactId> <version>1.1.1</version> </dependency> <dependency> <groupId>com.xiaomi</groupId> <artifactId>MiP

  • Java四种遍历Map的方法

    选择适合的最好 import java.util.HashMap; import java.util.Iterator; import java.util.Map; import java.util.Set; /** * Created by song on 2019/1/17. **/ public class MapT { public static void main(String[] args) { Map<Integer,String> map=new HashMap<>

  • 基于Java语言实现Socket通信的实例

    基于Java语言实现Socket通信 由于近日项目需求,需要在服务器中增加Socket通信的功能,接收硬件设备发送的心跳包和相关数据,因此又重新对Java的网络编程进行了复习,根据项目的实际情况做了简化的编程,实现了简单的通信过程. 1. Socket通信原理 源IP地址和目的IP地址以及源端口号和目的端口号的组合称为套接字.其用于标识客户端请求的服务器和服务. 以下是通过Socket套接字实现客户端与服务器端通信的示意图: 在实际应用中,客户端会通过访问服务器的IP和PORT连接到服务器端,这

  • Java算法实现调整数组顺序使奇数位于偶数之前的讲解

    调整数组顺序使奇数位于偶数之前 1. 题目描述 输入一个整数数组,实现一个函数来调整该数组中数字的顺序,使得所有的奇数位于数组的前半部分,所有的偶数位于位于数组的后半部分,并保证奇数和奇数,偶数和偶数之间的相对位置不变. 2. 题目分析 该题目类似于一个选择排序,将奇数选择出来,放置于数据前面的位置,保持其他未被选择的元素的相对位置不变: 1. 遍历数组,当数组元素为奇数是进行处理,判断条件为 n % 2 != 0 2. 设置一个变量标注当前已遍历的元素中奇数的个数oddNum,也是将该奇数元素

  • JavaScript变量提升和严格模式实例分析

    本文实例讲述了JavaScript变量提升和严格模式.分享给大家供大家参考,具体如下: 1.什么是变量提升 所谓的变量提升指的是:函数声明和变量声明总是会被解释器悄悄地被"提升"到方法体(作用域)的最顶部. 下面我们通过一个例子来详细说明一下. //变量声明在使用之前 var x; console.log(x);//undefined //变量声明在使用之后 console.log(y);//undefined var y; //上面的式子可以写成下面的样子 /* 变量提升:把x,y提

  • Java变态跳台阶实现思路和代码

    变态跳台阶 1. 题目描述 一只青蛙一次可以跳上1级台阶,也可以跳上2级--它也可以跳上n级.求该青蛙跳上一个n级的台阶总共有多少种跳法. 2. 题目分析 f(1) = 1 f(2) 会有两个跳得方式,一次1阶或者2阶,这回归到了问题f(1),f(2) = f(2-1) + f(2-2) f(3) 会有三种跳得方式,1阶.2阶.3阶,那么就是第一次跳出1阶后面剩下:f(3-1);第一次跳出2阶,剩下f(3-2):第一次3阶,那么剩下f(3-3).因此结论是: f(3) = f(3-1)+f(3-

  • 实例讲解Java中动态代理和反射机制

    反射机制 Java语言提供的一种基础功能,通过反射,我们可以操作这个类或对象,比如获取这个类中的方法.属性和构造方法等. 动态代理:分为JDK动态代理.cglib动态代理(spring中的动态代理). 静态代理 预先(编译期间)确定了代理者与被代理者之间的关系,也就是说,若代理类在程序运行前就已经存在了,这种情况就叫静态代理 动态代理 代理类在程序运行时创建的代理方式.也就是说,代理类并不是在Java代码中定义的,而是在运行期间根据我们在Java代码中的"指示"动态生成的. 动态代理比

随机推荐