博客
关于我
Java中Class对象详述
阅读量:90 次
发布时间:2019-02-26

本文共 3357 字,大约阅读时间需要 11 分钟。

理解Java中的运行时类型信息(RTTI)

运行时类型信息(RTTI)是Java编程中的一个核心概念,它允许在程序运行时发现和利用对象的具体类型信息。通过RTTI,我们可以在向上转型后,准确知道对象的实际类型,而无需直接操作派生类。


RTTI的作用

RTTI的主要目的是解决面向对象编程中的类型信息问题。在传统的面向对象编程中,我们常常需要知道一个对象的具体类型才能进行适当操作。然而,当我们将派生类的实例向上转型为基类时,直接操作基类引用会导致我们无法获取具体的运行时类型信息。


一个经典的RTTI示例

让我们来看一个经典的例子:

abstract class Vehicle {
abstract void run();
}
public class Car extends Vehicle {
@Override
void run() {
System.out.println("小汽车开动");
}
}
public class Bus extends Vehicle {
@Override
void run() {
System.out.println("公交车开动");
}
}
public class Bike extends Vehicle {
@Override
void run() {
System.out.println("自行车骑行");
}
}
public static void main(String[] args) {
// 均向上转型
Vehicle vehicle1 = new Car();
vehicle1.run(); // 输出:小汽车开动
Vehicle vehicle2 = new Bus();
vehicle2.run(); // 输出:公交车开动
Vehicle vehicle3 = new Bike();
vehicle3.run(); // 输出:自行车骑行
}

在这个代码中,我们无需直接调用派生类的方法。通过向上转型为基类Vehicle,我们可以使用同一个引用调用run方法。RTTI在这里发挥了关键作用,它告诉Java虚拟机(JVM)在运行时如何正确地将引用转换为具体的派生类实例。


向下转型的危险性

需要注意的是,向下转型(即从基类转换为派生类)是错误的操作。虽然编译时不会报错,但在运行时会抛出ClassCastException异常。

例如:

public class BYDCar extends Car {
@Override
void run() {
System.out.println("BYD小汽车开动");
}
}
public static void main(String[] args) {
// 向下转型
BYDCar bydCar = (BYDCar) new Car();
bydCar.run(); // 会抛出ClassCastException
}

运行时会报错:

Exception in thread "main" java.lang.ClassCastException: rtti.Car cannot be cast to rtti.BYDCar

这表明在向下转型时,Java会在运行时进行正确性检查,以确保我们的代码不会破坏类的继承结构。


Class对象的加载与获取

在Java中,每个类都有一个Class对象,用于创建和管理该类的实例。Class对象是动态加载的,具体过程如下:

  • 静态初始化块:当类第一次被加载时,static初始化块会执行。
  • 动态加载:当需要使用Class对象时,类加载器会检查是否已经加载该类。如果尚未加载,默认的类加载器会根据类名查找.class文件,并进行验证。
  • 通过A.classClass.forName("rtti.A")等方式,可以获取任意类的Class对象。


    获取Class对象的方法

  • Class.forName("类名字符串")

    传入一个类名字符串,返回对应的Class对象。需要处理可能的ClassNotFoundException异常。

  • 类名.class

    使用字面常量获取Class对象,这种方式更安全且简洁。

  • 实例对象.getClass()

    获取实例对象的Class对象引用。


  • Class.forName的使用

    public static void main(String[] args) {
    try {
    Class a = Class.forName("rtti.A");
    System.out.println(a); // 输出:class rtti.A
    } catch (ClassNotFoundException e) {
    System.out.println("Couldn't find A");
    }
    }

    Class对象的常用方法

    Class对象提供了丰富的方法来操作类信息,包括:

    • getName():返回类名。
    • getSimpleName():返回类名(不包含包名)。
    • getCanonicalName():返回规范化类名(包含包名)。
    • getFields():获取类的所有公有字段。
    • getMethods():获取类的所有公有方法。
    • getResource():获取类资源。

    实例分析

    让我们分析一个简单的Class对象使用示例:

    public class A {
    static {
    System.out.println("A is loaded");
    }
    int a = 0;
    int b = 1;
    public void say() {
    System.out.println("say");
    }
    public void hello() {
    System.out.println("hello");
    }
    }
    public static void main(String[] args) {
    try {
    Class clazz1 = Class.forName("rtti.A");
    System.out.println(clazz1.getName()); // 输出:rtti.A
    System.out.println(clazz1.getSimpleName()); // 输出:A
    System.out.println(clazz1.getCanonicalName()); // 输出:rtti.A
    for (Field f : clazz1.getDeclaredFields()) {
    System.out.println(f.getName());
    }
    for (Method m : clazz1.getMethods()) {
    System.out.println(m.getName());
    }
    } catch (ClassNotFoundException e) {
    System.out.println("Couldn't find A");
    }
    }

    最终总结

    RTTI是Java编程中的一个重要概念,它允许我们在运行时获取对象的具体类型信息。通过向上转型,我们可以避免直接操作派生类,从而实现代码的松耦合设计。理解Class对象的加载与操作,是掌握Java内省技巧的关键。

    如果你喜欢这篇文章,请关注我的公众号 Java技术秘籍,获取更多实用内容!

    转载地址:http://wxlk.baihongyu.com/

    你可能感兴趣的文章
    mysql与mem_细说 MySQL 之 MEM_ROOT
    查看>>
    MySQL与Oracle的数据迁移注意事项,另附转换工具链接
    查看>>
    mysql丢失更新问题
    查看>>
    MySQL两千万数据优化&迁移
    查看>>
    MySql中 delimiter 详解
    查看>>
    MYSQL中 find_in_set() 函数用法详解
    查看>>
    MySQL中auto_increment有什么作用?(IT枫斗者)
    查看>>
    MySQL中B+Tree索引原理
    查看>>
    mysql中cast() 和convert()的用法讲解
    查看>>
    mysql中datetime与timestamp类型有什么区别
    查看>>
    MySQL中DQL语言的执行顺序
    查看>>
    mysql中floor函数的作用是什么?
    查看>>
    MySQL中group by 与 order by 一起使用排序问题
    查看>>
    mysql中having的用法
    查看>>
    MySQL中interactive_timeout和wait_timeout的区别
    查看>>
    mysql中int、bigint、smallint 和 tinyint的区别、char和varchar的区别详细介绍
    查看>>
    mysql中json_extract的使用方法
    查看>>
    mysql中json_extract的使用方法
    查看>>
    mysql中kill掉所有锁表的进程
    查看>>
    mysql中like % %模糊查询
    查看>>