Object类介绍

Object类位于java.lang包,是所有Java类的根父类(唯一没有父类的类),所有类的直接或者间接父类。Java中的每个类都是由这个类扩展而来。

java.lang包在使用的时候无需显示导入,编译时由编译器自动导入。

Java认为所有的对象都具备一些基本的共性内容,这些内容可以不断的向上抽取,最终就抽取到了一个最顶层的类中的,该类中定义的就是所有对象都具备的功能。

如果在类的声明中未使用extends关键字指明其父类,则默认父类为Object类。 

public class Test [extends Object]{ // 默认继承了Object类
    //...
}

 

接收任意引用类型的对象

既然Object类是所有对象的父类,则肯定所有的对象都可以向Object进行转换,在这其中也包含了数组和接口类型,即:一切的引用数据类型都可以使用Object进行接收。

使用Object接收接口实例:

public class Test {
    public static void main(String[] args) {
        A ab = new B() ; // 为接口实例化
        Object obj = ab ; // 对象向上转型
        A a = (A)obj ; // 对象向下转型
        a.print();
    }
}
interface A { // 定义接口A
    public void print() ; // 定义抽象方法
}
class B implements A { // 子类实现接口
    public void print(){ // 覆写接口中的抽象方法
        System.out.println("test") ;
    }
}

使用Object接收数组:

public class Test {
    public static void main(String[] args) {
        int[] array = {1, 3, 5, 7, 9} ; // 定义数组
        Object obj = array ; // 使用Object接收数组
        print(obj) ; // 传递数组引用
    }
    public static void print(Object o){ // 接收一个对象
        if(o instanceof int[]) { // 判断对象的类型
            int x[] = (int[])o ; // 向下转型
            for(int i=0;i<x.length;i++){ // 循环输出
                System.out.print(x[i] + "\t") ;
            }
        }
    }
}

 

Object的主要方法

Object类中定义了一些有用的方法,由于是根类,所以这些方法都在其他类中存在,一般是进行了重载或者覆盖,实现了各自自己的具体功能。

 

public boolean equals(Object obj)

Object类中的equlas()方法用来检测两个对象是否等价,查看Object源码为:

public boolean equals(Object obj) {
    return (this == obj);
}

equals()只能比较引用类型,其作用与“==”相同,比较是否指向同一个对象(其实内部比较的就是两个对象地址)。对象比较的是引用,即对象在内存中的内存地址,而基本数据类型比较的是值。

用“==”进行比较时,符号两边的数据类型必须兼容(可自动转换的基本数据类型除外),否则编译出错。

 

equals()方法需要具有如下特点:

自反性(reflexive):任何非空引用x,x.equals(x)返回为true。

对称性(symmetric):任何非空引用x和y,x.equals(y)返回true当且仅当y.equals(x)返回true。

传递性(transitive):任何非空引用x和y,如果x.equals(y)返回true,并且y.equals(z)返回true,那么x.equals(z)返回true。

一致性(consistent):两个非空引用x和y,x.equals(y)的多次调用应该保持一致的结果,(前提条件是在多次比较之间没有修改x和y用于比较的相关信息)。

约定:对于任何非空引用x,x.equals(null)应该返回为false。

Student s1 = new Student(); 
Student s2 = new Student();
System.out.println(s1 == s2);   // false
Student s3 = s1;
System.out.println(s3 == s1);   // true
System.out.println(s1.equals(s2));  // false
System.out.println(s1.equals(s1));  // true
System.out.println(s1.equals(s3));  // true

当用equals()方法进行比较时,对类File、String、Date及包装类(Wrapper Class)来说,是比较类型及内容而不考虑引用的是否是同一个对象;因为在这些类中重写了Object类的equals()方法。下面就是String重写的equals()方法:

// 重写了Object类中的equlas()方法    
public boolean equals(Object anObject) {
    // 判断两个字符串的内存地址是否相等,如果相等,就说明这两个字符串是同一个
    if (this == anObject) {
        return true;
    }
    // 判断参数对象是不是String类型的
    if (anObject instanceof String) {
        // 如果是,就向上转型成一个String对象
        String anotherString = (String)anObject;
        // 对字符串逐个字符进行比较,每个字符都相等,返回true
        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;
}

 

public int hashCode()

返回该对象的哈希码值。哈希值是根据哈希算法计算出来的一个值。这个值和地址值有关,但不是实际地址值。

该方法用于哈希查找,可以减少在查找中使用equals的次数,重写了equals方法一般都要重写hashCode方法。这个方法在一些具有哈希功能的Collection中用到。

一般必须满足obj1.equals(obj2)==true,可以推出obj1.hashCode()==obj2.hashCode(),但是hashCode相等不一定就满足equals。不过为了提高效率,应该尽量使上面两个条件接近等价。如果不重写hashcode(),在HashSet中添加两个equals的对象,会将两个对象都加入进去。

// 对象不同,哈希值一般也不同
Student s1 = new Student();
System.out.println(s1.hashCode()); // 366712642
Student s2 = new Student();
System.out.println(s2.hashCode());  // 1829164700
System.out.println("-------");

 

public final Class getClass()

返回Object运行时类,也就是说是字节码文件对象。

不可重写,要调用的话,一般和getName()联合使用,如getClass().getName()。

Student s = new Student();
// getClass方法会返回一个Class类的实例对象,然后调用Class类的getName()方法返回全路径名
System.out.println(s.getClass().getName());  // com.zwj.objectdemo.Student

 

public String toString()

Object 类的 toString 方法返回一个字符串,该字符串由类名(对象是该类的一个实例)、at 标记符“@”和此对象哈希码的无符号十六进制表示组成。换句话说,该方法返回一个字符串,它的值等于: getClass().getName() + ‘@’ + Integer.toHexString(hashCode())

// toString方法:返回对象的字符串表示
System.out.println(s.toString());  // com.zwj.objectdemo.Student@7852e922
//直接输出对象的名称其实就是调用了对象的toString方法
System.out.println(s);   //com.zwj.objectdemo.Student@7852e922

 

protected void finalize()

当垃圾回收器确定不存在对该对象的更多引用时,由对象的垃圾回收器调用此方法,用于垃圾回收。但是具体什么时间调用此方法不确定。 

Java允许在类中定义一个名为finalize()的方法。它的工作原理是:一旦垃圾回收器准备好释放对象占用的存储空间,将首先调用其finalize()方法。并且在下一次垃圾回收动作发生时,才会真正回收对象占用的内存。

关于垃圾回收,有三点需要记住:

1)对象可能不被垃圾回收。只要程序没有濒临存储空间用完的那一刻,对象占用的空间就总也得不到释放。

2)垃圾回收并不等于“析构”。

3)垃圾回收只与内存有关。使用垃圾回收的唯一原因是为了回收程序不再使用的内存。

finalize()的用途:

无论对象是如何创建的,垃圾回收器都会负责释放对象占据的所有内存。这就将对finalize()的需求限制到一种特殊情况,即通过某种创建对象方式以外的方式为对象分配了存储空间。

不过这种情况一般发生在使用“本地方法”的情况下,本地方法是一种在Java中调用非Java代码的方式。

 

protected Object clone()

创建并返回此对象的一个副本,需要重写该方法。

要克隆的类去实现Cloneable接口 ,Cloneable接口是标记接口,实现该接口的类就可以实现对象的复制。

public class Student implements Cloneable{
    private String name; // 姓名
    
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    @Override
    protected Object clone() throws CloneNotSupportedException {
        return super.clone();
    }
}

我们来测试一下:

public class StudentTest{
    public static void main(String[] args) throws CloneNotSupportedException {
        // 创建显示对象
        Student s = new Student();
        s.setName("zwj");      
        // 克隆学生对象
        Student cloneStu = (Student) s.clone();
        // 引用
        Student s2 = s;
        System.out.println(s.getName());   // zwj,
        System.out.println(cloneStu.getName());  // zwj
        System.out.println(s2.getName());   // zwj

        s2.setName("爪哇菌");
        System.out.println(s.getName());   // 爪哇菌
        System.out.println(cloneStu.getName());  // zwj
        System.out.println(s2.getName());  // 爪哇菌
    }
}

由以上代码代码可以看出Clone()方法的使用比较简单,使用时注意如下几点即可:

什么时候使用shallow Clone,什么时候使用deep Clone,这个主要看具体对象的域是什么性质的,基本型别还是reference variable。

调用Clone()方法的对象所属的类必须实现(implements )Clonable接口,否则在调用Clone方法的时候会抛出CloneNotSupportedException。