IOS编程中对于==, isEqual, hash的理解【最详细分析】

前言
今天爱分享给大家带来IOS编程中对于==, isEqual, hash的理解【最详细分析】,并且自己建了demo给大家分析一下。

总结
先写下我看了这篇文章的总结:

1.==用于比较对象地址,如果不一致返回NO。
2.isEqual用于比较对象地址,但是可以重写,自定义判断逻辑。
3.hash不常用,在往NSSet添加对象时会用到。
4.3的具体逻辑是:在add时,系统会调用hash,如果返回值和集合里的一致,就会判断集合内已存在该对象,不给加。如果不一致,再调用isEqual方法判断。
如果要重写hash,直接return [super hash]是不对的。因为返回的是对象地址。这样会导致两个不同的对象永远加入不到同一个set里面。
例子
下面是一些测试例子,我创建了Person类,有name属性,我会重写isEqual和hash方法来测试。

1.==和isEqual方法,isEqual未重写

    Person *person1 = [[Person alloc] init];
    person1.name = @"1";
    
    Person *person2 = [[Person alloc] init];
    person2.name = @"1";
    
    NSLog(@"%d", person1==person1);
    NSLog(@"%d", person1==person2);
    NSLog(@"%d", [person1 isEqual:person1]);
    NSLog(@"%d", [person1 isEqual:person2]);

2020-03-07 14:06:48.004179+0800 AnimateDemo[43855:5522556] 1
2020-03-07 14:06:48.004259+0800 AnimateDemo[43855:5522556] 0
2020-03-07 14:06:48.004322+0800 AnimateDemo[43855:5522556] 1
2020-03-07 14:06:48.004391+0800 AnimateDemo[43855:5522556] 0

2.==和isEqual方法,isEqual重写

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

    Person *person1 = [[Person alloc] init];
    person1.name = @"1";
    
    Person *person2 = [[Person alloc] init];
    person2.name = @"1";
    
    NSLog(@"%d", person1==person1);
    NSLog(@"%d", person1==person2);
    NSLog(@"%d", [person1 isEqual:person1]);
    NSLog(@"%d", [person1 isEqual:person2]);

2020-03-07 14:23:33.606430+0800 AnimateDemo[43916:5530391] 1
2020-03-07 14:23:33.606498+0800 AnimateDemo[43916:5530391] 0
2020-03-07 14:23:33.606567+0800 AnimateDemo[43916:5530391] 1
2020-03-07 14:23:33.606639+0800 AnimateDemo[43916:5530391] 1

3.默认hash调用
我重写了hash的方法,然后NSSet里加对象

- (NSUInteger)hash {
    NSUInteger superHash = [super hash];
    NSLog(@"%ld", superHash);
    return superHash;
}

    Person *person1 = [[Person alloc] init];
    person1.name = @"1";
    
    Person *person2 = [[Person alloc] init];
    person2.name = @"1";
    
    NSMutableSet *set = [NSMutableSet set];
    [set addObject:person1];
    [set addObject:person2];
    NSLog(@"set count = %ld", set.count);

2020-03-07 14:28:55.113639+0800 AnimateDemo[43956:5534222] 105553148183088
2020-03-07 14:28:55.113703+0800 AnimateDemo[43956:5534222] 105553148183104
2020-03-07 14:28:55.113764+0800 AnimateDemo[43956:5534222] set count = 2

说明hash方法被调用了。
但是如果我不想将两个我认为一样的对象加入到set里面怎么办呢?比如例子中的两个person对象,我认为他们是一样的。
那么,重写hash并返回[super hash]是做不到的。因为不同对象的[super hash]返回的都不相同。
那么要怎么做呢?

4.正确的hash重写方法

- (NSUInteger)hash {
    NSUInteger nameHash = self.name.hash;
    NSLog(@"%ld", nameHash);
    return nameHash;
}

- (BOOL)isEqual:(id)object {
    if (self == object) {
        return YES;
    }

    if (![object isKindOfClass:[Person class]]) {
        return NO;
    }

    if ([((Person *)object).name isEqualToString:self.name]) {
        return YES;
    } else {
        return NO;
    }
}

    Person *person1 = [[Person alloc] init];
    person1.name = @"1";
    
    Person *person2 = [[Person alloc] init];
    person2.name = @"1";
    
    NSMutableSet *set = [NSMutableSet set];
    [set addObject:person1];
    [set addObject:person2];
    NSLog(@"set count = %ld", set.count);

2020-03-07 14:39:05.182269+0800 AnimateDemo[44016:5540023] 918
2020-03-07 14:39:05.182339+0800 AnimateDemo[44016:5540023] 918
2020-03-07 14:39:05.182412+0800 AnimateDemo[44016:5540023] set count = 1

正确的方法是用return self.name.hash;。
但是光有hash还不够,还需要重写isEqual方法。原因上面已经说了,set addObject方法会先去比较hash,在去比较isEqual。

人已赞赏
前端

Vue 实现当网址输入页面路由不存在时,页面跳转到404页面【附代码】

2020-12-23 14:42:36

IOS

iOS14 最新隐私适配 [相册]完美解决方案

2020-9-27 14:54:53

0 条回复 A文章作者 M管理员
    暂无讨论,说说你的看法吧
'); })();