Effective Java2读书笔记-对于所有对象都通用的方法(二)

时间:2022-12-03 16:01:50

第10条:始终要覆盖toString

这一条没什么好讲的,就是说默认的toString方法打印出来的是类名+@+十六进制哈希码的值。我们应该覆盖它,使它能够展示出一些更为详细清晰的信息,这个看实际情况吧。

第11条:谨慎地覆盖clone

有时候会出现这样的场景,你需要备份一些数据,对其一进行修改时,另外一个不受影响。这样,直接Foo a = new Foo(); Foo b = a; 是不可行的,b引用和a指向的是同一个对象。

这一条讲了很多,最后的总结是,Cloneable接口坑很多,因此最好别用。

当然,坑的原因就是因为类中包含引用类型的属性时,克隆时如果不做处理,就会克隆成引用。因此,如果只包含基本类型,还是可以用用的。典型用法如下

Effective Java2读书笔记-对于所有对象都通用的方法(二)Effective Java2读书笔记-对于所有对象都通用的方法(二)
public final class PhoneNumber implements Cloneable {
    private final short areaCode;
    private final short prefix;
    private final short lineNumber;

    public PhoneNumber(int areaCode, int prefix, int lineNumber) {
        rangeCheck(areaCode, 999, "area code");
        rangeCheck(prefix, 999, "prefix");
        rangeCheck(lineNumber, 9999, "line number");
        this.areaCode = (short) areaCode;
        this.prefix = (short) prefix;
        this.lineNumber = (short) lineNumber;
    }

    private static void rangeCheck(int arg, int max, String name) {
        if (arg < 0 || arg > max)
            throw new IllegalArgumentException(name + ": " + arg);
    }

    @Override
    public boolean equals(Object o) {
        if (o == this)
            return true;
        if (!(o instanceof PhoneNumber))
            return false;
        PhoneNumber pn = (PhoneNumber) o;
        return pn.lineNumber == lineNumber && pn.prefix == prefix
                && pn.areaCode == areaCode;
    }

    @Override
    public int hashCode() {
        int result = 17;
        result = 31 * result + areaCode;
        result = 31 * result + prefix;
        result = 31 * result + lineNumber;
        return result;
    }

    @Override
    public String toString() {
        return String.format("(%03d) %03d-%04d", areaCode, prefix, lineNumber);
    }

    @Override
    public PhoneNumber clone() {
        try {
            return (PhoneNumber) super.clone();
        } catch (CloneNotSupportedException e) {
            throw new AssertionError(); // Can't happen
        }
    }

    public static void main(String[] args) {
        PhoneNumber pn = new PhoneNumber(707, 867, 5309);
        Map<PhoneNumber, String> m = new HashMap<PhoneNumber, String>();
        m.put(pn, "Jenny");
        System.out.println(m.get(pn.clone()));
    }
}
View Code

就是实现Cloneable接口,然后覆盖Object类中的clone方法。首先调用super.clone,然后强转成当前类,再进行一些修正。例如数组,引用。

此时,也延伸一下数组的克隆。

第一种方法,再声明一个数组,遍历原数组,逐个填充进新数组(最low)。

第二种方法,调用数组的clone方法。

第三种方法(推荐),Arrays.copyOf(原始数组,新长度);