手记

Object类,hashCode(),equals(Object obj),finalize(),clone()

Object类
Object 是java中所有类的根类
所有类型直接或间接继承自Object类

构造方法:
Object()
学习常用的方法:
public int hashCode():返回该对象的哈希码值。
public final Class getClass():返回此 Object 的运行时类。
public String toString():返回该对象的字符串表示。建议所有子类都重写此方法。
getClass().getName() + '@' + Integer.toHexString(hashCode())
public boolean equals(Object obj):指示其他某个对象是否与此对象“相等”。
protected void finalize():当垃圾回收器确定不存在对该对象的更多引用时,由对象的垃圾回收器调用此方法。
protected Object clone():创建并返回此对象的一个副本。
hashCode()
hashCode()是从内存地址转换来的,代表了一个对象“的身份证号”
不同的对象调用hashCode()得出不同的哈希值
(对象的哈希值不同,就说明不是一个对象)
equals(Object obj)
Object类型原生的equal()方法代码如下:

public boolean equals(Object obj) {
     return (this == obj);// 比较内存地址值,换句话说,默认情况下,只能是在堆上,是同一个内存地址时,才会返回true
}
/**
 * 给自定义类型Student,重写equals()方法,让Student的对象
 * 进行“相等”比较,让对象之间具有比较功能
 */
@Override
public int hashCode() {
      // TODO Auto-generated method stub
      int i = super.hashCode();
      return (int) (i * 31 + System.currentTimeMillis());
}
@Override
public boolean equals(Object obj) {
      // return super.equals(obj);
      // 我们要对相比所有的成员变量是否
      // Student stu = (Student) obj;// ClassCastException
      // 优化程序的执行效率,先判断地址值,如果是同一块内存地址,就直接返回true
      if (this == obj) {
            return true;
      }
      // instanceof 判断一个对象,是否是 某一个类型的实例,来避免向下类型转换时的异常ClassCastException
      if (obj instanceof Student) {
            Student stu = (Student) obj;
            // (str1.equals(str2))
            if (this.getName().equals(stu.getName()) && this.getAge() == stu.getAge() && this.getBirth().equals(stu.getBirth())) {
                return true;
            }
      }
      return false;
}

finalize()

protected void finalize() throws Throwable {
    // 比如,这里打开了文件,(外部资源)
    // gc 的特点,不是立马执行的,我们直接执行finalize()方法,不能直接调用gc来清除内存
    super.finalize();
}

clone()

Object objclone = stu.clone();
// 对象的clone,之后强制类型转换
Student stuclone = (Student) objclone;
stu.setName("小可爱");// 改变了基本数据的值,不会影响克隆后的值
stu.changDate();// 改变了引用类型的值,会影响克隆后的值,也就说,两个应用类型指向了同一个对象
// 现在的问题是:stu 和 stuclone 之间是什么关系?
// 回答:stu 和stuclone 克隆关系,地址值不一样,成员变量一样
// 这种copy 我们通常叫 shadow clone,浅克隆,浅拷贝
// 还有与之对应的 叫 深拷贝,深度克隆,deep clone
// 深度克隆的意思是:在工作中,有可能需要 引用类型的值,也同时需要copy一份
// 默认JDK 不提供深度克隆,如果需要深度克隆,需要工程师自己想办法实现之

总结来说,浅拷贝 是copy 成员变量的时候,只copy基本数据类型
(和String类型,因为String类型是一种特殊的引用类型,
在这里可以看作“常量”,因为JDK设计者,将String类型设计为“不可变”的字符串);
引用数据类型不copy(引用类型 所指向对象,不再克隆,只保留一份,相当于两个变量指向了同一个对象)

Cloneable 接口,没有抽象方法,空接口,在jdk中还有很多,这种接口,标记接口

// 重写clone()方法
@Override
protected Object clone() throws CloneNotSupportedException {
      // TODO Auto-generated method stub
      // Student stu = (Student) super.clone();
      // if(stu.getAddress() != null){
      // stu.setAddress(null);
      // }
      // return super.clone();
      // return super.clone();
      Student stu = (Student) super.clone();// 只执行浅克隆
      // 如果要执行深度克隆,我们可以逐个 对引用类型 进行 浅克隆
      if (stu.birth != null) {
            Object obj = this.birth.clone();
            Date d = (Date) obj;
            stu.setBirth(d);
      }
      if (stu.address != null) {
            Object obj = this.address.clone();
            Address ad = (Address) obj;
            stu.setAddress(ad);
      }
      return stu;
}

对象的克隆(浅克隆和深克隆的示意图)

2人推荐
随时随地看视频
慕课网APP