Java基础教程(19)--Object类

??Object类位于类结构树的最顶端,所有的类都是它的直接或间接子类,因此所有的类都继承了Object类的方法,我们可以在需要的时候覆盖这些方法。下面是一些将会在本文中讨论的Object类的方法:

  • protected Object clone() throws CloneNotSupportedException

    ??创建并返回此对象的副本。

  • public boolean equals(Object obj)

    ??判断某个对象是否与这个对象“相等”。

  • protected void finalize() throws Throwable

    ??当垃圾回收器将对象从内存中清理出去之前要做的清理工作。

  • public final Class getClass()

    ??返回对象所属的类类型。

  • public int hashCode()

    ??返回对象的hash值。

  • public String toString()

    ??返回对象的字符串表示形式。

??下面的notify,notifyAll和wait方法在同步独立运行的线程的活动中扮演着不同的角色,本文不会去介绍它们,有关这一部分的内容将会在以后的文章中讨论:

  • public final void notify()
  • public final void notifyAll()
  • public final void wait()
  • public final void wait(long timeout)
  • public final void wait(long timeout, int nanos)

一.equals方法

??Object了类中的equals方法用于检测一个对象是否等于另外一个对象。在Object类中,这个方法将会判断两个对象是否具有相同的引用。如果两个对象具有相同的引用,它们一定是相等的。从这点上看,将其作为默认操作也是合乎情理的。然而,对于大多数类来说,这种判断并没有什么意义。我们在判断两个对象是否相等时,应该比较它们的内容,而不仅仅是判断它们是不是同一个对象。因此,大多数情况下,当我们需要使用equals方法时,都应该对它进行重写。

??为了演示,我们首先编写一个Apple类:

public class Apple {
    private String color;
    private int weight;
    public void setColor(String color) {
        this.color = color;
    }
    public String getColor() {
        return color;
    }
    public void setWeight(int weight) {
        this.weight = weight;
    }
    public int getWeight() {
        return weight;
    }
}

??当两个苹果的重量和颜色一样时,我们就认为它们是相等的。因此,Apple类的equals方法可以这么写:

public boolean equals(Object obj) {
    if (obj == null) {
        return false;
    }
    if (this == obj) {
        return true;
    }
    if (this.getClass() != obj.getClass()) {
        return false;
    }
    Apple apple = (Apple) obj;
    return weight == apple.getWeight() && Objects.equals(color, apple.getColor());
}

??为了防备color为null的情况,上面的例子中使用了Objects.equals(Object a,Object b)方法。如果a和b都是null,这个方法将会返回true;如果其中只有一个参数为null,则返回false;如果a和b都不为null,则会返回a.equals(b)的结果。

??实际上,上面的equals还存在一定的问题。不过本文属于基础教程系列,因此不会深入讲解这其中的问题。有兴趣的读者可以查阅Java规范中对于equals方法的要求以及参考其他深入讨论equals方法的文章。

二.hashCode方法

??hashCode方法的返回值是根据对象本身所计算出来的散列值(也称哈希值)。如果两个对象是相等的,那么对它们调用hashCode方法得到的返回值也应该是相等的。如果重写了equals方法,那么默认的hashCode方法也不再适用。因此,如果重写了equals方法,则必须同时重写hashCode方法。在重写hashCode方法时,原则上只需要保证两个相等的对象的散列值是相同的即可。不过如何减少冲突以及编写更高效的哈希函数,可以参考其他文章或查阅计算机算法书中关于哈希的内容。下面编写了一个Apple类的hashCode方法作为示例:

public int hashCode() {
    return 7 * (color == null ? 0 : color.hashCode()) + 11 * weight;
}

三.clone方法

??如果一个类或它的某个超类实现了Cloneable接口,那么就可以使用clone()方法从这个类的实例上创建一个副本。在调用clone()方法时,编译器会检查这个类是否实现了Cloneable接口。如果没有,编译器将会抛出一个CloneNotSupportedException异常。有关异常的内容会在后面的文章中介绍,现在你只需要知道要覆盖clone()方法,必须将它声明为:

protected Object clone() throws CloneNotSupportedException

public Object clone() throws CloneNotSupportedException

??如果调用clone方法的对象实现了Cloneable接口,则继承自Object类的clone()方法将会创建与原始对象相等的对象,使其具有与原始对象的相应成员变量相同的值。因此,如果想要让类可以clone,只需要将implements Cloneable添加到类的声明中即可。

??对于某些类,Objects类的clone方法可以正常工作。但是,如果对象包含对外部对象的引用,则可能需要覆盖clone方法。否则,即使克隆的对象与元对象不是一个对象,但它们内部引用的还是相同的对象。这样一来,对内部对象所做的更改也会影响到另一个对象。如果需要克隆出一个完全与原对象隔离的新对象,则需要重写clone方法,将每个内部对象再拷贝一次。

四.finalize方法

??finalize方法用于定义在回收对象前要执行的清理工作。Object类的finalize方法什么也没做,只有一个空方法体,可以覆盖finalize方法来定义清理行为,例如释放资源等。finalize方法不需要也不建议手动调用,它会在垃圾回收器回收对象时自动调用。

五.toString方法

??toString方法用于返回表示对象值的字符串。为每个类提供toString方法是一个良好的习惯。

??下面是Object类的toString方法:

public String toString() {
    return getClass().getName() + "@" + Integer.toHexString(hashCode());
}

??可以看到,Object类的toString方法返回的是类名加对象的hashCode的十六进制表示,中间使用符号@隔开。不过在打印对象的信息时,这个方法的返回值并没有什么意义。因此,建议在编写的每一个类中都覆盖toString方法。例如为上面的Apple类编写toString方法:

public String toString() {
    return getClass().getName() + "[color = " + color + ",weight = " + weight + "]";
}

六.getClass方法

??Class类是一个表示类的信息的类。对对象调用getClass方法会返回一个Class类的实例,用来表示当前对象所属对象的信息。由于getClass方法是final的,因此无法对它进行重写。

??Class类提供了非常多的方法,例如获取类名的方法getSimpleName(),获取父类的方法geuSuperClass(),获取实现的接口的方法getInterfaces()等。例如,下面的方法会打印出对象的类名:

void printClassName(Object obj) {
    System.out.println("The object‘s" + " class is " + obj.getClass().getSimpleName());
}

??有关Class的内容会在后面有关反射的文章中进行介绍,这里只需要知道getClass方法的作用即可。

原文地址:https://www.cnblogs.com/maconn/p/10259560.html

时间: 2024-08-29 19:19:38

Java基础教程(19)--Object类的相关文章

java基础学习总结——Object类

永不放弃,一切皆有可能!!! 只为成功找方法,不为失败找借口! java基础学习总结——Object类 一.Object类介绍 Object类在JAVA里面是一个比较特殊的类,JAVA只支持单继承,子类只能从一个父类来继承,如果父类又是从另外一个父类继承过来,那他也只能有一个父类,父类再有父类,那也只能有一个,JAVA为了组织这个类组织得比较方便,它提供了一个最根上的类,相当于所有的类都是从这个类继承,这个类就叫Object.所以Object类是所有JAVA类的根基类,是所有JAVA类的老祖宗.

[ 转载 ] Java基础10--关于Object类下所有方法的简单解析

关于Object类下所有方法的简单解析 类Object是类层次结构的根类,是每一个类的父类,所有的对象包括数组,String,Integer等包装类,所以了解Object是很有必要的,话不多说,我们直接来看jdk的源码,开始我们的分析之路 1.hashcode() public native int hashCode();//native说明跟机器有关,跟对象的地址有关 如果我们新建一个类,而hashcode没有被重写的话,那么hashcode返回的值只于对象的地址有关,如果hashcode被重

Java基础10:object类

Object类 这部分主要参考http://ihenu.iteye.com/blog/2233249 Object类是Java中其他所有类的祖先,没有Object类Java面向对象无从谈起.作为其他所有类的基类,Object具有哪些属性和行为,是Java语言设计背后的思维体现. Object类位于java.lang包中,java.lang包包含着Java最基础和核心的类,在编译时会自动导入.Object类没有定义属性,一共有13个方法,13个方法之中并不是所有方法都是子类可访问的,一共有9个方法

Java基础—API及Object类和String类

Java 的API(API: Application(应用) Programming(程序) Interface(接口)) Java API就是JDK中提供给我们使用的类,这些类将底层的代码实现封装了起来,我们不需要关心这些类是如何实现的,只需要学习这些类如何使用即可. 1.Object类  Object类是Java语言中的根类,即所有类的父类.它中描述的所有方法子类都可以使用.所有类在创建对象的时候,最终找的父类就是Object.   equals方法    equals方法,用于比较两个对象

Java基础小常识-Object类-(15)

一.演示实例1 /* Object:所有类的根类. Object是不断抽取而来,具备着所有对象都具备的共性内容. 常用的共性功能: *//*class Fu // extends Object{    Fu()    {        super();    }}class Zi extends Fu{    Zi()    {        super();    }}*/ class Person extends Object{    private int age;    Person(i

Java基础教程:面向对象编程

Java基础教程:面向对象编程 Java语言概述 Java语言特点 1.Java为纯面向对象的语言,它能够直接反映现实生活中的对象.总之,Everything is object! 2.平台无关性.Java为解释型语言,编译器会把Java代码变成"""中间代码",然后在JVM上解释执行. 3.Java提供了很多内置的类库,这些类库简化了开发人员的程序设计工作,同时缩短了项目开发时间. 4.Java语言提供了对Web应用的支持. 5.Java语言提供了较好的安全性和健

Java基础教程:JDBC编程

Java基础教程:JDBC编程 快速开始 什么是JDBC JDBC 指 Java 数据库连接,是一种标准Java应用编程接口( JAVA API),用来连接 Java 编程语言和广泛的数据库. JDBC API 库包含下面提到的每个任务,都是与数据库相关的常用用法. 制作到数据库的连接. 创建 SQL 或 MySQL 语句. 执行 SQL 或 MySQL 查询数据库. 查看和修改所产生的记录. 从根本上来说,JDBC 是一种规范,它提供了一套完整的接口,允许便携式访问到底层数据库,因此可以用 J

Java基础教程:多线程基础(2)——线程间的通信

Java基础教程:多线程基础(2)--线程间的通信 使线程间进行通信后,系统之间的交互性会更强大,在大大提高CPU利用率的同时还会使程序员对各线程任务在处理的过程中进行有效的把控与监督. 线程间的通信 思维导图 等待中 等待/通知机制 不使用等待/通知机制 我们可以使用使用sleep()与 whle(true) 死循环来实现多个线程间的通信. 虽然两个线程实现了通信,但是线程B必须不断的通过while语句轮训机制来检测某一个条件,这样会浪费CPU资源. 如果轮询间隔较小,更浪费时间间隔.如果轮训

Java基础教程(25)--I/O流

??I/O流表示输入源或输出目标.流可以表示许多不同类型的源和目标,例如磁盘文件.设备.其他程序等. ??流支持许多不同类型的数据,包括字节.原始数据类型.字符和对象等.有些流只传递数据; 有些流则可以操纵和转换数据. ??无论各种流的内部是如何工作的,所有流都提供相同的简单模型:流是一系列数据.程序使用输入流从源头获取数据,一次一项: ??程序使用输出流将数据写入目的地,一次一项: ??在本文中,我们会看到流可以处理各种各样的数据,无论是基本数据还是复杂对象.先来几张IO流的全家福: ??In