Java的Object类

简介: Object 类是 Java 中所有类的始祖,在 Java 中每个类都是由它扩展而来的。

Object 类

Object 类是 Java 中所有类的始祖,在 Java 中每个类都是由它扩展而来的。但是并不需要这样写:public class Employee extends Object 如果没有明确地指出父类,Object 就被认为是这个类的父类。

所有的数组类型,不管是对象数组还是基本类型的数组都扩展了 Object 类。

Employee[] staff = new Employee[10];
obj = staff; // OK
obj = new int[10]; // OK

equals() 方法

Object 类中的 equals() 方法用于检测一个对象是否等于另外一个对象。在 Object 类中,equals() 方法将判断两个对象是否具有相同的引用。然而,我们经常需要检测两个对象状态的相等性,如果两个对象的状态相等,就认为这两个对象
是相等的。


为了防备 a、b 变量可能为 null 的情况,需要使用 Objects.equals() 方法。

  • 如果两个参数都为 null,Objects.equals(a, b) 调用将返回 true;
  • 如果其中一个参数为 null,Objects.equals(a, b) 调用将返回 false;
  • 如果两个参数都不为 null,则调用 a.equals(b)。

Java 语言规范要求 equals() 方法具有下面的特性:

  • 自反性:对于任何非空引用 x,x.equals(x) 应该返回 true。
  • 对称性:对于任何引用 x 和 y,当且仅当 y.equals(x) 返回 true,x.equals(y) 也应该返回 true。
  • 传递性:对于任何引用 x、y 和 z,如果 x.equals(y) 返后 true,y.equals(z) 返回 true,x.equals(z) 也应该返回 true。
  • 一致性:如果 x 和 y 引用的对象没有发生变化,反复调用 x.equals(y) 应该返回同样的结果。
  • 对于任意非空引用 x,x.equals(null) 应该返回 false。

这些规则十分合乎情理,从而避免了类库实现者在数据结构中定位一个元素时还要考虑调用 x.equals(y),还是调用 y.equals(x) 的问题。

相等测试

下面给出编写一个完美的 equals() 方法的建议:

  1. 显式参数命名为 otherObject,稍后需要将它转换成另一个叫做 other 的变量。
  2. 检测 this 与 otherObject 是否引用同一个对象:if (this == otherObject) return true;这条语句只是一个优化。实际上,这是一种经常采用的形式。因为计算这个等式要比一个一个地比较类中的域所付出的代价小得多。
  3. 检测 otherObject 是否为 null,如果为 null,返回 false。if (otherObject = null) return false;这项检测是很必要的,避免后面判断 otherObject 的实例域时出现 NullPointerException 异常。
  4. 比较 this 与 otherObject 是否属于同一个类:

    • 如果 equals 的语义在每个子类中有所改变,就使用 getClass 检测:if (getClass() != otherObject.getClass()) return false;
    • 如果所有的子类都拥有统一的语义,就使用 instanceof 检测:if (!(otherObject instanceof ClassName)) return false;
  5. 将 otherObject 转换为相应的类的类型变量:ClassName other = (ClassName) otherObject;
  6. 现在开始对所有需要比较的域进行比较了。使用 == 比较基本类型域,使用 equals() 方法比较对象域。如果所有的域都匹配,就返回 true;否则返回 false。return fieldl == other.field && Objects.equa1s(fie1d2, other.field2)

如果在子类中重新定义 equals() 方法,就要在其中调用父类的 equals() 方法 super.equals(other)。如果检测失败,对象就不可能相等。如果父类中的域都相等,就需要比较子类中的实例域。

提示:对于数组类型的域,可以使用静态的 Arrays.equals() 方法检测相应的数组元素是否相等。


// String 类的 equals() 方法
public boolean equals(Object anObject) {
    // 检测 this 与 anObject 是否引用同一个对象
    if (this == anObject) {
        return true;
    }
    if (anObject instanceof String) {
        // 将 otherObject 转换为相应的类的类型变量
        String anotherString = (String) anObject;
        // 对所有需要比较的域进行比较
        int n = value.length;
        if (n == anotherString.value.length) {
            char v1[] = value;
            char v2[] = anotherString.value;
            int i = 0;
            while (n-- != 0) {
                if (v1[i] != v2[i])
                    return false;
                i++;
            }
            return true;
        }
    }
    return false;
}

hashCode() 方法

如果重新定义 equals() 方法,就必须重新定义 hashCode() 方法,以便用户可以将对象插入到散列表(Map 集合)中。

equals() 与 hashCode() 的定义必须一致:如果 x.equals(y) 返回 true,那么 x.hashCode() 就必须与 y.hashCode() 具有相同的值。例如,如果用定义的 Employee.equals() 比较雇员的 ID,那么 hashCode() 方法就需要散列 ID,而不是雇员的姓名或存储地址。

提示:如果存在数组类型的域,那么可以使用静态的 Arrays.hashCode() 方法计算一个散列码,这个散列码由数组元素的散列码组成。

// String 类的 hashCode() 方法
public int hashCode() {
    int h = hash;
    if (h == 0 && value.length > 0) {
        char val[] = value;

        for (int i = 0; i < value.length; i++) {
            h = 31 * h + val[i];
        }
        hash = h;
    }
    return h;
}

toString() 方法

在 Object 类中还有一个重要的方法,就是 toString() 方法,它用于返回表示对象值的字符串。


随处可见 toString() 方法的主要原因是:只要对象与一个字符串通过 “+” 操作符连接起来,Java 编译就会自动地调用 toString() 方法,以便获得这个对象的字符串描述。

// java 文件
public static void main(String[] args) {
    P p = new P();
    String s = "p =" + p;
}

// class 文件
public static void main(String[] args) {
    P p = new P();
    String s = "p =" + String.valueOf(p);
}

如果 x 是任意一个对象, 并调用 System.out.println(x); println() 方法就会直接地调用 x.toString(),并打印输出得到的字符串。

Object 类定义了 toString() 方法,用来打印输出对象所属的类名和散列码。例如,调用 System.out.println(System.out)
将输出下列内容:java.io.PrintStream@2f6684。

要想打印数组,调用静态方法 Arrays.toString() 方法。要想打印多维数组(即,数组的数组)则需要调用 Arrays.deepToString() 方法。

// Object 类的 toString() 方法
public String toString() {
    return getClass().getName() + "@" + Integer.toHexString(hashCode());
}

参考资料

《Java核心技术卷一:基础知识》(第10版)第 5 章:继承 5.2 Object:所有类的超类

相关文章
|
28天前
|
存储 缓存 安全
java 中操作字符串都有哪些类,它们之间有什么区别
Java中操作字符串的类主要有String、StringBuilder和StringBuffer。String是不可变的,每次操作都会生成新对象;StringBuilder和StringBuffer都是可变的,但StringBuilder是非线程安全的,而StringBuffer是线程安全的,因此性能略低。
44 8
|
28天前
|
Java 开发者
在 Java 中,一个类可以实现多个接口吗?
这是 Java 面向对象编程的一个重要特性,它提供了极大的灵活性和扩展性。
61 1
|
1月前
|
存储 安全 Java
java.util的Collections类
Collections 类位于 java.util 包下,提供了许多有用的对象和方法,来简化java中集合的创建、处理和多线程管理。掌握此类将非常有助于提升开发效率和维护代码的简洁性,同时对于程序的稳定性和安全性有大有帮助。
63 17
|
1月前
|
安全 Java
Java多线程集合类
本文介绍了Java中线程安全的问题及解决方案。通过示例代码展示了使用`CopyOnWriteArrayList`、`CopyOnWriteArraySet`和`ConcurrentHashMap`来解决多线程环境下集合操作的线程安全问题。这些类通过不同的机制确保了线程安全,提高了并发性能。
|
1月前
|
存储 Java 程序员
Java基础的灵魂——Object类方法详解(社招面试不踩坑)
本文介绍了Java中`Object`类的几个重要方法,包括`toString`、`equals`、`hashCode`、`finalize`、`clone`、`getClass`、`notify`和`wait`。这些方法是面试中的常考点,掌握它们有助于理解Java对象的行为和实现多线程编程。作者通过具体示例和应用场景,详细解析了每个方法的作用和重写技巧,帮助读者更好地应对面试和技术开发。
117 4
|
1月前
|
Java 编译器 开发者
Java异常处理的最佳实践,涵盖理解异常类体系、选择合适的异常类型、提供详细异常信息、合理使用try-catch和finally语句、使用try-with-resources、记录异常信息等方面
本文探讨了Java异常处理的最佳实践,涵盖理解异常类体系、选择合适的异常类型、提供详细异常信息、合理使用try-catch和finally语句、使用try-with-resources、记录异常信息等方面,帮助开发者提高代码质量和程序的健壮性。
62 2
|
1月前
|
存储 安全 Java
如何保证 Java 类文件的安全性?
Java类文件的安全性可以通过多种方式保障,如使用数字签名验证类文件的完整性和来源,利用安全管理器和安全策略限制类文件的权限,以及通过加密技术保护类文件在传输过程中的安全。
53 4
|
1月前
|
Java 数据格式 索引
使用 Java 字节码工具检查类文件完整性的原理是什么
Java字节码工具通过解析和分析类文件的字节码,检查其结构和内容是否符合Java虚拟机规范,确保类文件的完整性和合法性,防止恶意代码或损坏的类文件影响程序运行。
44 5
|
1月前
|
Java API Maven
如何使用 Java 字节码工具检查类文件的完整性
本文介绍如何利用Java字节码工具来检测类文件的完整性和有效性,确保类文件未被篡改或损坏,适用于开发和维护阶段的代码质量控制。
79 5
|
1月前
|
存储 Java 编译器
java wrapper是什么类
【10月更文挑战第16天】
41 3