NSDictionary实现原理-ios哈希hash和isEqual

 
OC中自定义类的NSCopying实现的注意事项(isEqual & hash实现)
 
iOS开发 之 不要告诉我你真的懂isEqual与hash!

NSDictionary(字典)是使用 hash表来实现key和value之间的映射和存储的, hash函数设计的好坏影响着数据的查找访问效率。数据在hash表中分布的越均匀,其访问效率越高。而在Objective-C中,通常都是利用NSString 来作为键值,其内部使用的hash函数也是通过使用 NSString对象作为键值来保证数据的各个节点在hash表中均匀分布。

见NSDictionary中最常用的一个方法原型:

[objc] view plain copy

 
  1. - (void)setObject:(id)anObject forKey:(id <NSCopying>)aKey;
从这个方法中可以知道, 要作为 Key 值,必须遵循 NSCopying 协议。这是因为在NSDictionary内部,会对 aKey 对象 copy 一份新的。而  anObject 对象在其内部是作为强引用(retain或strong)。
 

既然知道了作为 key 值,必须遵循 NSCopying 协议,说明除了 NSString 对象之外,我们还可以使用其他类型对象来作为 NSDictionary 的 key值。不过这还不够,作为 key 值,该类型还必须继承于 NSObject 并且要重载一下两个方法:

[objc] view plain copy

 
  1. - (NSUInteger)hash;
 
[objc] view plain copy

 
  1. - (BOOL)isEqual:(id)object;
 
 
其中,hash 方法是用来计算该对象的 hash 值,最终的 hash 值决定了该对象在 hash 表中存储的位置。我们重写hash方法是因为每向NSDictionary和NSSet中存入一个key-value,字典会先利用即将插入的key的hash和字典中已经存在的所有的key.hash进行比较,最终来决定是新增一个key,还是覆盖原有的key。 但是仅仅使用key.hash比较,有时会出现2个对象hash相同的情况,这时候就需要调用isEqual 方法来最终裁定,2个key对象是否相同。
 
 
 

在OC中,如果自定义类,则要考虑赋值、持久化保存、保存到其它容器中等各种情况的对象复制和比较,下面是一个比较全面的自定义例子,在此仅作记录:

自定义类:

KeyValuePairs.h:

[objc] view plain copy

 
  1. #import <Foundation/Foundation.h>
  2. @interface KeyValuePairs: NSObject <NSCopying>
  3. @property (nonatomic,strong)NSString *identifier;
  4. @property (nonatomic,strong)NSString *name;
  5. @end
KeyValuePairs.m:
[objc] view plain copy

 
  1. #import "KeyValuePairs.h"
  2. @implementation KeyValuePairs
  3. - (id)copyWithZone:(NSZone *)zone
  4. {
  5. KeyValuePairs *kvp = [[[self class] allocWithZone:zone] init];
  6. kvp.identifier = self.identifier;
  7. kvp.name = self.name;
  8. return kvp;
  9. }
  10. - (BOOL)isEqualToKeyValuePairs:(KeyValuePairs *)kvp{
  11. if (!kvp) {
  12. return NO;
  13. }
  14. BOOL haveEqualName = (!self.name && !kvp.name) || [self.name isEqualToString:kvp.name];
  15. BOOL haveEqualIdentifier = (!self.identifier && !kvp.identifier) || [self.identifier isEqualToString:kvp.identifier];
  16. return haveEqualName && haveEqualIdentifier;
  17. }
  18. #pragma mark -NSObject
  19. -(BOOL)isEqual:(id)object{
  20. if (self == object) {
  21. return YES;
  22. }
  23. if (![object isKindOfClass:[KeyValuePairs class]]) {
  24. return NO;
  25. }
  26. return [self isEqualToKeyValuePairs:(KeyValuePairs *)object];
  27. }
  28. - (NSUInteger)hash {
  29. return [self.name hash] ^ [self.identifier hash];
  30. }
  31. @end

测试:

[objc] view plain copy

 
  1. NSMutableDictionary *namesWillUpdateDic = [[NSMutableDictionary alloc] init];
  2. NSMutableArray *names = [[NSMutableArray alloc] init];
  3. for (int i = 0; i<1000; i++) {
  4. NSString *name = [NSString stringWithFormat:@"%d_zhangsan",i];
  5. NSString *identifier = [NSString stringWithFormat:@"%d_identifier",i];
  6. NSString *strObj = [NSString stringWithFormat:@"%d_strObj",i];
  7. KeyValuePairs *kvp = [[KeyValuePairs alloc] init];
  8. kvp.identifier = identifier;
  9. kvp.name = name;
  10. [namesWillUpdateDic setObject:strObj forKey:kvp];
  11. [names addObject:kvp];
  12. }
  13. for (int j = 0; j<1000; j++) {
  14. int index = arc4random()%1000;
  15. KeyValuePairs *kvp = [names objectAtIndex:index];
  16. NSString *strObj = [namesWillUpdateDic objectForKey:kvp];
  17. NSString *msg = [NSString stringWithFormat:@"index:%d,identifier:%@,email:%@,strObj:%@",index,kvp.identifier,kvp.name,strObj];
  18. NSLog(@"%@",msg);
  19. }
 
 
 
注意:
1、自定义类为什么一定要实现NSCopying协议呢?这是因为通过key-value把2个对象加入到字典中,字典会对key进行copy一份的操作,而对value对象进行retain操作,如果自定义的类不实现copy协议,那么就不能作为字典的key对象使用。
2、如果自定义类不重写isEqual则默认使用内存地址比较两个对象,可能会出现意想不到的结果
3、isEqual和hash方法要同时重写,否则isEqual方法判断将不正确
 
 
 

NSDictionary实现原理-ios哈希hash和isEqual的更多相关文章

  1. iOS判断对象相等 重写isEqual、isEqualToClass、hash

    相等的概念是探究哲学和数学的核心,并且对道德.公正和公共政策的问题有着深远的影响. 从一个经验主义者的角度来看,两个物体不能依据一些观测标准中分辨出来,它们就是相等的.在人文方面,平等主义者认为相等意 ...

  2. redis 哈希(hash)函数

    哈希(hash)函数 hSet 命令/方法/函数 Adds a value to the hash stored at key. If this value is already in the has ...

  3. redist命令操作(二)--哈希Hash,列表List

    1.Redis 哈希(Hash) 参考菜鸟教程:http://www.runoob.com/redis/redis-hashes.html Redis hash 是一个string类型的field和v ...

  4. Redis中的哈希(Hash)

    Redis 哈希(Hash) Redis hash 是一个string类型的field和value的映射表,hash特别适合用于存储对象. Redis 中每个 hash 可以存储 232 - 1 键值 ...

  5. 大话Java中的哈希(hash)结构(一)

    o( ̄▽ ̄)d 小伙伴们在上网或者搞程序设计的时候,总是会听到关于“哈希(hash)”的一些东西.比如哈希算法.哈希表等等的名词,那么什么是hash呢? 一.相关概念 1.hash算法:一类特殊的算法 ...

  6. Python操作redis系列以 哈希(Hash)命令详解(四)

    # -*- coding: utf-8 -*- import redis #这个redis不能用,请根据自己的需要修改 r =redis.Redis(host=") 1. Hset 命令用于 ...

  7. Redis 命令,键(key),字符串(String),哈希(Hash),列表(List),集合(Set)(二)

      Redis 命令 Redis 命令用于在 redis 服务上执行操作. 要在 redis 服务上执行命令需要一个 redis 客户端.Redis 客户端在我们之前下载的的 redis 的安装包中. ...

  8. 哈希--Hash,“散列”/“哈希”

    哈希 Hash,翻译“散列”,音译为“哈希”,把任意长度的输入,通过散列算法,变换成固定长度的输出,该输出就是散列值.这种转换是一种压缩映射,也就是散列值的空间通常远小于输入的空间,不同的输入可能会散 ...

  9. 区块链 - 哈希(Hash)

    章节 区块链 – 介绍 区块链 – 发展历史 区块链 – 比特币 区块链 – 应用发展阶段 区块链 – 非对称加密 区块链 – 哈希(Hash) 区块链 – 挖矿 区块链 – 链接区块 区块链 – 工 ...

随机推荐

  1. java post请求的表单提交和json提交简单小结

    在java实现http请求时有分为多种参数的传递方式,以下给出通过form表单提交和json提交的参数传递方式: public String POST_FORM(String url, Map< ...

  2. 第三十四天- 线程队列、线程池(map/submit/shutdown/回调函数)

    1.线程列队 queue队列 :使用import queue,用法与进程Queue一样 class queue.Queue(maxsize=0) # 先进先出: q = queue.Queue(3) ...

  3. (转)预处器的对比——Sass、LESS和Stylus

    英文原文:http://net.tutsplus.com/tutorials/html-css-techniques/sass-vs-less-vs-stylus-a-preprocessor-sho ...

  4. 纯小白入手 vue3.0 CLI - 2.4 - 新组件 Forms.vue 中学习表单

    vue3.0 CLI 真小白一步一步入手全教程系列:https://www.cnblogs.com/ndos/category/1295752.html 我的 github 地址 - vue3.0St ...

  5. cordova app强制横屏

    非常简单,只需要在config.xml里加上这行: <preference name="Orientation" value="landscape" /& ...

  6. JavaScript实现IP地址的输入框方式

    最近遇到一些这样的需求:实现一种IP地址的输入方式,就是输入3个字符或自动跳到下一个输入框内,删除的时候,一个输入框没有了字符,自动跳回上一个输入框.看到这里,相信大家都有一些想法了,没错,这种方法就 ...

  7. [iOS] UICollectionView实现图片水平滚动

    最新更新: 简单封装了一下代码,参考新文章:UICollectionView实现图片水平滚动 先简单看一下效果: 新博客:http://wossoneri.github.io 准备数据 首先先加入一些 ...

  8. using 和try/catch区别和注意点

    书上解释: using: 在C#和其他托管语言中,没有自动.决定性的析构方式,而是有一个垃圾收集器,它会在未来的某个时刻释放资源.它是非决定性的,因为我们不能确定这个过程在什么时候发生.忘记关闭数据库 ...

  9. Lodash学习笔记

    有多年开发经验的工程师,往往都会有自己的一套工具库,称为utils.helpers等等,这套库一方面是自己的技术积累,另一方面也是对某项技术的扩展,领先于技术规范的制订和实现. Lodash就是这样的 ...

  10. VMWare12虚拟机实现主客机间的文件拖拽(复制粘贴)和文件夹共享

    版本: 主机:Windows 7 64位旗舰版 虚拟机: VMWare 12 + Windows 7 64位旗舰版 VMWare pro 12 + Ubuntu16.04LTS 64位 注:由于VMW ...