在 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);}
}

本文首发于java黑洞网,csdn同步更新

如果有人再问你 Java 的反射,把这篇文章扔给他相关推荐

  1. 如果有人再问你 Java IO,把这篇文章砸他头上

    一.简介 说到 I/O,想必大家都不会陌生, I/O 英语全称:Input/Output,即输入/输出,通常指数据在内部存储器和外部存储器或其他周边设备之间的输入和输出. 比如我们常用的 SD 卡.U ...

  2. java说影_面试被问到Java虚拟机,用这篇文章怼过去

    Jvm内存结构 Jvm内存结构,一般是面试官对Java虚拟机这块考察的第一问.讲真,还没背会,自己罚自己面壁思过. Java虚拟机的内存结构一般可以从线程共有和线程私有两部分起头作答,然后再详细说明各 ...

  3. 面试官,不要再问我“Java虚拟机类加载机制”了

    关于Java虚拟机类加载机制往往有两方面的面试题:根据程序判断输出结果和讲讲虚拟机类加载机制的流程.其实这两类题本质上都是考察面试者对Java虚拟机类加载机制的了解. 面试题试水 现在有这样一道判断程 ...

  4. 不要再问我“Java GC垃圾回收机制”了

    点击蓝色"程序猿DD"关注我 回复"资源"获取独家整理的学习资料! Java GC垃圾回收几乎是面试必问的JVM问题之一,本篇文章带领大家了解Java GC的底 ...

  5. serve注解是加在哪个类_不会 Java 注解 ? 看这一篇文章!

    对于Java注解,我之前的印象是很模糊的,总觉得这个东西经常听说,也经常用,但是具体是怎么回事,好像没有仔细学习过,说到注解,立马想到@Controller,仅此而已. 对于Java注解,我咨询过一些 ...

  6. java怎么运行_不要再问我Java程序是怎么执行的了!

    什么是Java虚拟机? 要弄明白Java程序的执行过程首先要了解一下Java虚拟机. 虚拟机是一种抽象化的计算机,通过在实际的计算机上仿真模拟各种计算机功能来实现的.Java虚拟机有自己完善的硬体架构 ...

  7. 【面试必读】求你们不要再问我Java中的锁及优化了?

    本文90%内容整理自掘金:https://juejin.im/post/5c8a00e56fb9a049a42feb54#heading-27 一.Java线程阻塞的代价 Java的线程是映射到操作系 ...

  8. IPTV是什么意思-2020年还有人再问IPTV是什么

    IPTV是伴随着计算机网络的发展而出现的新型观看电视方式,传统电视观看是有线电视,而随着网络的发展,现在使用IPTV系统看电视似乎已经占据了主流,尤其是在国内IPTV系统是运营商配合家庭宽带一起销售, ...

  9. java getinstance 反射_Java 基础篇之反射

    使用反射获取程序运行时的对象和类的真实信息. 获取 Class 对象 每个类被加载之后,系统会为该类生成一个对应的 Class 对象,通过该 Class 对象可以访问到 JVM 中的这个类. 使用 C ...

最新文章

  1. Activity一共有以下四种launchMode
  2. Xamarin Anroid开发教程之下载安装Xamarin
  3. 有用过PHP SPL的吗,不知道这主要用来干嘛
  4. CodeForces 1131G. Most Dangerous Shark
  5. composer设置代理_composer 设置代理
  6. [vue] v-on可以绑定多个方法吗?
  7. 5G iPhone SE或将在明年一季度推出 明年有望生产3000万部
  8. 【Clickhouse】Clickhouse 访问控制和账号管理
  9. 拯救普通人周报焦虑,一个自动化报表工具就能实现
  10. 无需Docker, 5分钟徒手DIY 一个Linux容器
  11. 年薪 80w 的程序员被鄙视了!
  12. 如何将图片转换成PCBLogo
  13. PHP自学笔记 ---李炎恢老师PHP第一季 TestGuest0.4
  14. java 图片实现毛玻璃_iOS毛玻璃效果的实现及图片模糊效果的三种方法
  15. mysql lag over_lag函数(lag函数用法)
  16. 51单片机之外部中断拙见
  17. 智能红外遥控器(二): Arduino环境搭建及库安装
  18. 收发一体超声波测距离传感器模块_大禹电子教你如何区别超声波传感器、超声波探头和超声波换能器...
  19. 企业实施erp系统前企业应该了解的常识
  20. 3.基础实验-实验01-点亮一个LED灯

热门文章

  1. linux内核--设备驱动程序(学习笔记)
  2. MySQL slowlog 统计_mysql slow log 简单统计
  3. c post请求网页_Python使用urllib2抓取网页
  4. (40)System Verilog线程停止(disable fork)
  5. ZYNQ PS端输出无效时钟供PL使用
  6. stm32f103c8t6 AD DMA连续采集8个通道
  7. as5300g2 nas软件功能_群晖NAS软件Qnote介绍及使用方法教程
  8. 我从创立3家科技公司的经历中学到了什么
  9. Python--turtle绘图模块讲解
  10. 学计算机不会重装系统正常吗,系统重装不了的原因是什么 重装不了系统的解决方法【图文】...