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

当我们创建了作者对象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);

// 输出:com.cmower.java_demo.fifteen.Writer@7852e922

} catch (InstantiationException | IllegalAccessException e1) {

e1.printStackTrace();

}

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

Class 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();

}

四、反射

====

我们还可以通过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 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) {

架构学习资料

准备两个月,面试五分钟,Java中高级岗面试为何越来越难?

准备两个月,面试五分钟,Java中高级岗面试为何越来越难?

准备两个月,面试五分钟,Java中高级岗面试为何越来越难?

准备两个月,面试五分钟,Java中高级岗面试为何越来越难?

准备两个月,面试五分钟,Java中高级岗面试为何越来越难?

由于篇幅限制小编,pdf文档的详解资料太全面,细节内容实在太多啦,所以只把部分知识点截图出来粗略的介绍,每个小节点里面都有更细化的内容!
G-1719268963739)]

[外链图片转存中…(img-5JB6nBMf-1719268963739)]

[外链图片转存中…(img-s3YB7fI5-1719268963740)]

[外链图片转存中…(img-a8fP7YB2-1719268963740)]

[外链图片转存中…(img-s7f8dVcq-1719268963741)]

由于篇幅限制小编,pdf文档的详解资料太全面,细节内容实在太多啦,所以只把部分知识点截图出来粗略的介绍,每个小节点里面都有更细化的内容!

  • 16
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值