Java如何在运行时识别类型信息?

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

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

一、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对象知道某个对象“真正”的类型,并不会因为向上转型而丢失。

二、获取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语句块中。

三、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);
    // 输出:[email protected]
} 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);
    // 输出:[email protected]
} catch (InstantiationException | IllegalAccessException e1) {
    e1.printStackTrace();
}

四、反射

我们还可以通过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虚拟机的角度,还是程序员的角度?”
他给了我很好的解释和启发,我不由觉得非常的惭愧,作为一名年纪颇长的Java学习者,竟然对理论知识薄弱到令人发指的地步——不知道你是否也有这样的困惑?
但写作的好处就在于此,在向读者解释“Java如何在运行时识别类型信息”的过程中,我的思路逐渐地清晰了起来——这真是一个自我提升的好办法!

原文地址:https://blog.51cto.com/13754022/2373817

时间: 2024-11-03 21:33:02

Java如何在运行时识别类型信息?的相关文章

java编译时与运行时概念与实例详解 -------------------(*************************)

Java编译时与运行时很重要的概念,但是一直没有明晰,这次专门博客写明白概念. 基础概念 编译时  编译时顾名思义就是正在编译的时候.那啥叫编译呢?就是编译器帮你把源代码翻译成机器能识别的代码.(当然只是一般意义上这么说,实际上可能只是翻译成某个中间状态的语言.比如Java只有JVM识别的字节码,.另外还有啥链接器.汇编器.为了了便于理解我们可以统称为编译器) 那编译时就是简单的作一些翻译工作,比如检查老兄你有没有粗心写错啥关键字了啊.有啥词法分析,语法分析之类的过程.就像个老师检查学生的作文中

Java编程思想——第14章 类型信息(二)反射

六.反射:运行时的类信息 我们已经知道了,在编译时,编译器必须知道所有要通过RTTI来处理的类.而反射提供了一种机制——用来检查可用的方法,并返回方法名.区别就在于RTTI是处理已知类的,而反射用于处理未知类.Class类与java.lang.reflect类库一起对反射概念进行支持,该类库包含Field.Method以及Constructor(每个类都实现了Member接口).这些类型是由JVM运行时创建的,用来表示未知类种对应的成员.使用Constructor(构造函数)创建新的对象,用ge

java 反射机制:运行时的类信息(为框架服务的Bug存在)

反射机制:JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法:对于任意一个对象,都能够调用它的任意一个方法:这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制. 换一种引出反射的说法是:当通过反射与一个未知的类型的对象打交道是,JVM只是简单地检查这个类,看它是属于哪个特定的类(就想RTTI那样).在用它做其他事情之前必须先加载那个类的Class对象.因此,那个类的.class文件对于JVM来说必须是可获取的:那么在本地机器上,要么通过网络获得

深入理解Java虚拟机:运行时数据区域

Java虚拟机在执行Java程序的过程中会把它所管理的内存划分为若干个不同的数据区域.这些区域都有各自的用途,以及创建和销毁的时间,有的区域随着虚拟机进程的启动而存在,有些区域则依赖用户线程的启动和结束而建立和销毁.根据<Java虚拟机规范(Java SE 7版)>的规定,Java虚拟机所管理的内存将会包括以下几个运行时数据区域. 程序计数器 程序计数器(Program Counter Register)是一块较小的内存空间,它可以看作是当前线程所执行的字节码的行号指示器.在虚拟机的概念模型里

java内存结构(运行时数据区域)

java虚拟机规范规定的java虚拟机内存其实就是java虚拟机运行时数据区,其架构如下: 其中方法区和堆是由所有线程共享的数据区. Java虚拟机栈,本地方法栈和程序计数器是线程隔离的数据区. (1).程序计数器: 是一块较小的内存空间,其作用可以看作是当前线程所执行的字节码的行号指示器,字节码解析器工作时通过改变程序计数器的值来选取下一条需要执行的字节码指令.程序的分支.循环.跳转.异常处理以及线程恢复等基础功能都是依赖程序计数器来完成. Java虚拟机的多线程是通过线程轮流切换并分配处理器

深入理解Java虚拟机笔记---运行时栈帧结构

栈帧(Stack Frame)是用于支持虚拟机进行方法调用和方法执行的数据结构,它是虚拟机运行时数据区的虚拟机栈(Virtual Machine Stack)的栈元素.栈帧存储了方法的局部变量表,操作数栈,动态连接和方法返回地址等信息.第一个方法从调用开始到执行完成,就对应着一个栈帧在虚拟机栈中从入栈到出栈的过程. 每一个栈帧都包括了局部变量表,操作数栈,动态连接,方法返回地址和一些额外的附加信息.在编译代码的时候,栈帧中需要多大的局部变量表,多深的操作数栈都已经完全确定了,并且写入到了方法表的

深入JAVA虚拟机之运行时数据区

前言最近在啃一本书<深入JAVA虚拟机>,这本书不是第一次看,可以说是从大学就开始看,这一次应该算第三次啃这本书,也应该说算是第一次真正啃这本书.大学的时候,只是好奇表层的一些神奇现象,随着工作几年后,现在回过头来再次啃这本书,对于表层的那些以前觉得神奇的现在已经感觉乏味,反而对于底层是如何实现.如何运作的越来越着迷.这也是这次看这本书的初衷.通过写博客记录下自己的学习过程,也方便以后回头看看现在的看法想法在将来会变成怎样.如果我在下面的文字表述上或者理解上有误解或者错误,请各位大神能够留言指

java虚拟机2.运行时内存对象

对象的创建 虚拟机遇到一条new指令时,首先去检查这个指令的参数是否能在常量池中定位到一个类的符号引用,并且检查这个符号引用代表的类是否已被加载.解析和初始化过. 在类加载检查通过后,接下来虚拟机将为新生对象分配内存.对象所需内存的大小在类加载完成后便可完全确定,为对象分配空间的任务等同于把一块确定大小的内存从java堆中划分出来. 假设java堆中内存是绝对规整的,所有用过的内存都放在一边,空闲的内存放在另一边,中间放着一个指针作为分界点的指示器,那分配内存就仅仅是把那个指针向空闲空间那边挪动

java虚拟机3.运行时内存异常

在java虚拟机规范的描述中,除了程序计数器外,虚拟机内存的其他几个运行时区域都有发生OutOfMemoryError异常的可能. java堆溢出 java堆用于存储对象实例,只要不断的创建对象,并且保证GC Roots到对象之间有可达路径来避免垃圾回收机制清除这些对象,那么在对象数量达到最大堆得容量限制后就会产生内存溢出异常.当出现java堆内存溢出时,异常堆栈信息"java.lang.OutOfMemoryError"会跟着进一步提示"java heap space&qu