高效编写代码的方法(六):判断对象相等

isEqual

比较对象是否相等,在OC中目前有两种方式:
==isEqual:方法。
对于==方法,对于指针类对象的判断,其实挺鸡肋的,系统只是简单的进行指针地址的对比。当我们想更“深入”地对比对象的时候,显然==是不能满足要求的。
对于我们在框架中常用的类型对象,比如NSString,NSArray类等,系统对于这些类,已经实现了一个isEqual方法可以帮助我们去对对象做比较。对于NSString类,还有一个特性的方法:

- (BOOL)isEqualToString:(NSString *)aString;

由此增加这个相等方法对于类型的判断。提高对比效率,我们在自己实现isEqualTo方法是也要加强类型判断。

下面两个方法是比较对象时最为关键的两个方法:

- (BOOL)isEqual:(id)object;
- (NSUInteger)hash;

默认的isEqual实现与==类似,都是直接比较指针指向地址是否相同。
我们在自己重写isEqual时也应该重写hash方法,由此保证,相同的两个对象返回的hash是一样的。但是有相同hash的对象,却又不一定的相等的。

下面是一个重写isEqual的例子:

- (BOOL)isEqual:(id)object
{
    if (self == object) {
        return YES;
    }
    if ([self class] != [object class]) {
        return NO;
    }
    Teacher *anotherTeacher = (Teacher *)object;
    if (![self.name isEqualToString:anotherTeacher.name]) {
        return NO;
    }
    if (![self.workNumber isEqualToString:anotherTeacher.workNumber]) {
        return NO;
    }
    return YES;
}

以上是一个Teacher类的对比方法,首先对比指针,如果self 和object相同,那么必定相等。然后我们在做类型判断,如果两者类型都不相等,那么必定不相等。但是如果有一个类叫JohnTeacher,固定name为john,是Teacher的子类,此时如果一个Teacher的name属性为john,那么两者是可能相等的,此时在isEqual方法实现中,就要对类型的判断做更加细致的判断,建议使用isKindOfClass进行判断。最后就是进行对象的各属性是否相等的判断了,如果属性都相等,即可认为两个对象相等。

为保证相同对象都有相同的hash值这样一个规定,重写hash也是必须的。
比如可以这样:

-(NSUInteger)hash{
    return 1337;
}

这样来说,相同的类的对象都有同一个Hash。

但是在一个集合使用的场景下,比如向一个NSSet(不允许对象重复)中添加对象。那个这个set在添加对象的时候,就会先去对比Hash值,发现hash相同,那么会再去调用isEqual去进一步判断对象是否相等,如果添加的对象过多,那么每一次都会调用isEqual,造成执行效率低。

另一种hash的实现方法如下:

-(NSUInteger)hash{
    NSString *stringToHash = [NSString stringWithFormat:@"%@:%@",_name,_workNumber];
    return [stringToHash hash];
}

对由对象属性拼接而成的字符串进行一次hash计算,最后返回这个值,由此可以确保相同对象,hash值一定是一样的。但是这种方法肯定不如返回规定值来的快。

最后总结一个最合适的方法:

-(NSUInteger)hash{
    NSUInteger nameHash = [_name hash];
    NSUInteger workNumberHash = [_workNumber hash];
    return nameHash ^ workNumberHash;
}

以上是一种相对折中的方法,hash值不会单一,执行量也不算太大。

规定类型的判断方法

对于NSString来说,isEqualToString肯定要比isEqual方法来的更易读,类型的规定也更加的明确。
所以我们也可以实现自己的强类型判断的方法:

- (BOOL)isEqualToTeacher:(Teacher *)anotherTeacher
{
    if (self == anotherTeacher) {
        return YES;
    }
    if (![self.name isEqualToString:anotherTeacher.name]) {
        return NO;
    }
    if (![self.workNumber isEqualToString:anotherTeacher.workNumber]) {
        return NO;
    }
    return YES;
}

如此,在isEqual方法中,我们也可以写的更简单一点:

- (BOOL)isEqual:(id)object
{
    if ([self class] == [object class]) {
        return [self isEqualToTeacher:object];
    }else{
        return [super isEqual:object];
    }
}

深相等与浅相等

以上isEqual中我们都对比了对象的每一个属性是否相等,这样的一种详尽的对比方式可以称为“深相等”

但是某些时候,比如Teacher,我们可以给他分配一个标识属性,比如identifier。这个identifier对外只读,并且每个对象identifier不相同,由此我们可以直接对比identifier即可,该方法为“浅相等”。

可变对象的对比

下面这个例子,中文有点难以描述,大致意思是可变对象的hash会随对象的改变而改变,将对象加入一个集合中时可能会引发问题。直接上代码:

 NSMutableSet *set = [[NSMutableSet alloc] init];
    NSMutableArray *arrayA = [@[@1,@2]  mutableCopy];
    [set addObject:arrayA];
    NSLog(@"1: set = %@", set);
    
    NSMutableArray *arrayB = [@[@1,@2]  mutableCopy];
    [set addObject:arrayB];
    NSLog(@"2: set = %@", set);
    
    NSMutableArray *arrayC = [@[@1]  mutableCopy];
    [set addObject:arrayC];
    NSLog(@"3: set = %@",set);
    [arrayC addObject:@2];
    NSLog(@"4: set = %@",set);
    
    NSSet *setB = [set copy];
    NSLog(@"5: setB = %@",setB);

整理后的打印结果:


LogResult

可以观察到第四个输出结果set = {(1,2),(1,2)},显然与Set的定义相违背(不能有相同对象),而且更奇怪的是setB 拷贝了set后变成了{(1,2)}
所以我们在遇到这样的情况时,不要修改被加入集合后的对象,否则导致像以上代码那样的混乱。

总结

  • 1 如果想自定义对比对象的方法,重写isEqual和hash方法。
  • 2 相同对象,hash必须相同,但拥有相同hash的对象不一定本身相同。
  • 3 有选择的对比对象属性,而不是全部对比(针对有特殊标识属性的对象)
  • 4 hash方法要可读并高效
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 151,511评论 1 330
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 64,495评论 1 273
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 101,595评论 0 225
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 42,558评论 0 190
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 50,715评论 3 270
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 39,672评论 1 192
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,112评论 2 291
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 29,837评论 0 181
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 33,417评论 0 228
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 29,928评论 2 232
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 31,316评论 1 242
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 27,773评论 2 234
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 32,253评论 3 220
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 25,827评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,440评论 0 180
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 34,523评论 2 249
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 34,583评论 2 249

推荐阅读更多精彩内容