第3章.对于所有对象都通用的方法[Effective Java]

这里说的通用方法 , 可能更多指Object中的可被重写的方法

一. equals 方法

1. 重写equals时请准守通用约定

  • 自反性 ( reflexive )
x.equals(x) == true
  • 对称性 ( symmetric )
x.equals(y) == y.equals(x)
  • 传递性 ( transitive )
if ( x.equals(y) && y.equals(z) ){
        x.equals(z) == true;
}
  • 一致性 ( consistent )
    有对象 x , y . 如果 x.equals(y) == true 那么 , 只要 x , y 没有被修改就必须保证 x.equals(y) 永远都是 true . ( 若一开始是 false 则一直是 false )
  • x.equals(null) == false

2. 实现 equals 方法的诀窍

  • 使用 == 操作符检查 " 参数是否为正确的类型 "
  • 使用 instanceof 操作符检查 " 参数是否为正确的类型 "
  • 把参数转换成正确的类型
  • 对于类中的每个关键域 , 检查参数中的域和该对象对应域是否匹配
public class User{//此处省略getter和setter
        private String name;
        private String age;
        public boolean equals(Object o){
                if ( o == this ){
                    return true;
                }
                if (!(o instanceof User)){
                      return false;
                }
                User user = (User)o;
                return equalsStr(user.name,name)
                     &&equalsStr(user.age,age);
        }
        public boolean equalsStr(String str1,String str2){
                if ( str1 == null ){
                      return str1 == str2;
                }else {
                      return str1.equals(str2)
                }
        }
}

二. hashCode 方法 ( equals 方法被重写的时候 , 此方法也总是需要被重写 )

1. Object.hashCode通用约定

  • 程序执行期间 , equals 中所用到的域值没有被修改 , 则hashCode方法的返回值不变 . (程序重启后可能可上一次程序运行时的值不一样)
  • 若两对象 x.equals(y) == true 则 x.hashCode() == y.hashCode()
  • 若两对象 x.equals(y) == false 则 x.hashCode() != y.hashCode()

2. 相对理想的hashCode实现方法

private boolean isRight;
private byte b;
private long l;
private float f;
private double d;
//假设在equals方法中用到了以上关键域

private volatile int hashCode;
public int hashCode(){
    int result = hashCode;//可以是任意非0常量
    if(result != 0){
        return result;//当计算复杂的时候 , 缓存hashCode
    }
    int[] keys = new int[5];//上面有5个关键域
    keys[0] = isRight ? 1:0;
    keys[1] = (int)b;
    keys[2] = (int)(l ^ ( l >>> 32 ));
    keys[3] = Float.floatToIntBits(f);
    long ld = Double.doubleToLongBits(d);
    keys[4] = (int)(ld ^ (ld >>> 32 ));
    for (int key : keys){
        result = 31*result+key;
    }
    return result;
}

三. toString方法

Object类中toString方法的默认实现 :

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

此方法返回的字符串足够简洁 , 但信息不够丰富 , 因此 " 建议所有的子类都重写这个方法 "

  • 考虑指定toString方法返回数据的格式 ( 如 : json , xml 等)

指定返回格式后 , 可以提供一个静态工厂或者构造器允许传入符合格式的字符串作为参数来创建对象 . 这样 , 对象和字符串就可以相互转换

  • 指定返回格式后 , 要始终保持一致 , 避免后续因为格式变动引发的错误
  • 为返回字符串中包含对象的所有信息 ( 这些信息需要可被外部访问或者有可被外部访问的公有方法 )

四. clone方法

此章节主要介绍深克隆和浅克隆 . 一个较为合适的实现方法 , 应该是深克隆

1. 当类成员均引用的不可变对象

public final class PhoneNumber implements Cloneable{//省略getter和setter
    private final short areaCode;
    private final short prefix;
    private final short lineNumber;
    public PhoneNumber clone() throws CloneNotSupportedException{
        return (PhoneNumber)super.clone();
    }
}

因为PhoneNumber类中所有的成员变量引用的都是不可变对象 , 所以以上方法可以实现深克隆

2. 当类成员引用了"深层结构"的可变对象

当类成员中有可变对象时 , 若简单的使用 super.clone() 得到新的克隆实例 , 此时修改新实例中此可变成员的值 , 会导致原实例中此成员值一并修改 ( 新实例和原实例的成员变量指向同一个实例 )

此时应该先调用super.clone()的到一个新的实例 , 然后修正任何需要修正的成员变量值 ( final 修饰的成员变量无效 )

public final class PhoneNumber implements Cloneable{//省略getter和setter
    private final short areaCode;
    private final short prefix;
    private final short lineNumber;
    private Object[] elements;
    public PhoneNumber clone() throws CloneNotSupportedException{
        PhoneNumber phoneNumber = (PhoneNumber)super.clone();
        phoneNumber.elements = elements.clone();//此类中仅有此域需要修正
        return phoneNumber;
    }
}

3. 更好的办法实现对象拷贝

使用上面的方法可能过于复杂 , 可以考虑提供拷贝构造器或者拷贝工厂

public final class PhoneNumber{//省略getter和setter
    private final short areaCode;
    private final short prefix;
    private final short lineNumber;
    private Object[] elements;
    private String s;
    public PhoneNumber(PhoneNumber phoneNumber){
        this.areaCode = phoneNumber.areaCode;
        this.prefix = phoneNumber.prefix;
        this.lineNumber = phoneNumber.lineNumber;
        this.elements = phoneNumber.elements.clone();
        this.s = new String(phoneNumber.s);//保证新的实例中所有成员指向的实例都有原来的不一致
    }
    public static PhoneNumber(PhoneNumber phoneNumber){
        return new PhoneNumber(phoneNumber);
    }
}

五. 考虑实现Comparable接口

compareTo方法并没有在Object中声明 , 但是实现它可以获得强大的功能 .

符号sgn ( x ) 表示数学中的signum的函数 , 它根据 x 的值为负值 、零和正值分别返回 -1 , 0 , 1 .

接口实现约定

  • sgn(x.compareTo(y)) == -sgn(y.compareTo(x))
  • 比较关系传递 , 若 ( x.compareTo(y) > 0 && y.compareTo(c) ) 则 x.compareTo(z) > 0
  • 强烈建议 (x.compareTo(y) == 0) == x.equals(y)
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,390评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,821评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,632评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,170评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,033评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,098评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,511评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,204评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,479评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,572评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,341评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,213评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,576评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,893评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,171评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,486评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,676评论 2 335

推荐阅读更多精彩内容