如何理解iOS的“对象等同性”
在iOS开发过程中,我们经常需要用到等同性来判断两个对象是否相等,通常我们会使用==来判断,但是这样比较出来的结果可能不是我们期望的;所以,一般我们会使用NSObject协议声明的isEqual方法来判断对象的等同性。并且,为了更好的进行深层次的比较,iOS系统中的NSObject子类还实现了各自的isEqual:方法。
== 究竟比较的是什么?
对于基本类型,==比较的是值;对于对象类型,`==``比较的是对象的地址,即是否为同一个对象
NSString *s1 = @"123";
NSString *s2 = [NSString stringWithFormat:@"%d", 123];
BOOL e1 = s1 == s2;
BOOL e2 = [s1 isEqual:s2];
BOOL e3 = [s1 isEqualToString:s2];
NSLog(@"%d, %d, %d", e1, e2, e3); // 0, 1, 1
从上面的例子可以看出,由于s1和s2不是同一个对象(即对象地址不同),所以==结果为0(NO),而isEqual和isEqualToString方法则判断对象是否相同,所以结果为:1、1。
如何重写isEqual方法
NSObject协议中有两个方法用于判断等同性:
- (BOOL)isEqual:(id)object;
@property (readonly) NSUInteger hash;
这两个方法的默认实现是:当且仅当其“指针值”完全相同时,这两个对象才相等。如果isEqual判断两个对象相等,那么其hash值一定相等;反之,如果两个对象的hash值相等,则对象不一定相等(原因:hash值的获取方式可能造成冲突,导致尽管hash的key值不同,但是hash值是一样)。
iOS系统已经实现了部分NSObject子类的isEqual方法(更多参考Equality),如:
1. NSString - isEqualToString
2. NSArray - isEqualToArray
3. NSDictionary - isEqualToDictionary
4. NSSet - isEqualToSet
但是对于自定义的类型来说,如果有对象等同性的比较需求,那么需要自行实现isEqual方法,具体步骤如下:
.1 实现一个isEqualTo__ClassName__: 方法来执行有意义的值比较
.2 重写isEqual:方法来作类型和对象等同性检查, 回调上述的值比较方法
.3 重写 hash, 在集合中查找时最先调用
实例代码:
Person头文件
@interface Person : NSObject
@property (nonatomic, copy) NSString *name;
@property (nonatomic, strong) NSDate *birthday;
@end
Person实现isEqual方法
- (BOOL)isEqual:(id)object {
if (self == object) {
return YES;
}
if (![object isKindOfClass:[Person class]]) {
return NO;
}
return [self isEqualToPerson:(Person *)object];
}
- (BOOL)isEqualToPerson:(Person *)person {
if (!person) {
return NO;
}
BOOL haveEqualNames = (!self.name && !person.name) || [self.name isEqualToString:person.name];
BOOL haveEqualBirthdays = (!self.birthday && !person.birthday) || [self.birthday isEqualToDate:person.birthday];
return haveEqualNames && haveEqualBirthdays;
}
上述代码实现了比较自定义对象的等同性,也适用于具有继承关系的场景
实现hash方法
在实现hash方法之前,需要先了解一下hash表是什么:hash表也称散列表,或哈希表,hash表是一种特殊的数据结构,它同数组、链表以及二叉排序树等相比较有很明显的区别,它能够快速定位到想要查找的记录,而不是与表中存在的记录的关键字进行比较来进行查找。这个源于hash表设计的特殊性,它采用了函数映射的思想将记录的存储位置与记录的关键字关联起来,从而能够很快速地进行查找。
本质来讲,就是把所有成员的固定值(也可以是转换后的固定值)通过f(x)映射后形成的一张表,然后在需要查找成员时,就直接利用hash表来找,不需要顺序查找或链式查找,速度最快可以达到O(1)的级别,是典型的空间换时间的做法。
hash方法只有在被添加到NSSet和设置为NSDictionary的key时才会被调用,这是因为NSSet需要根据hash值来快速查找成员,而NSDictionary在查找key时,也利用key的hash查找来提高查找效率。
hash方法与判等的关系?
hash是对象等同性的必要非充分条件,在NSSet和NSDictionary中判断时,会先判断hash值是否相等,如果相等,那么就会进行isEqual的判断;反之,不相等,直接判断对象不相等
重写hash方法:
- (NSUInteger)hash {
return [self.name hash] ^ [self.birthday hash];
// NSObject的hash值是调用hash方法的对象地址,一般不用,需要重写一个hash的方法实现
//return [super hash];
}
上述是hash值的一种实现方式,用属性的XOR方式来设置唯一的hash值,可以满足绝大多数的场景需求。其中,hash值的实现有多种不同的方式,能够产生唯一性的hash值的概率越高,表明hash的可靠性越高。一般情况下,hash值都是唯一的,利于快速查找;但是,如果hash值出现相等的情况,即出现冲突,那么就需要特殊处理,具体的处理方法请参考文末的资料。
参考资料
iOS判断对象相等 重写isEqual、isEqualToClass、hash
iOS开发 之 不要告诉我你真的懂isEqual与hash!
如何理解iOS的“对象等同性”的更多相关文章
- Effective Objective-C 2.0 — 第8条:理解“对象等同性”这一概念
第8条:理解“对象等同性”这一概念 若想检测对象的等同性,请提供“isEqual”与 hash 方法 相同的对象必须具有相同哈希码,但是两个哈希码相同的对象却未必相同. 不要盲目地逐个检测每条属性,而 ...
- 0112.1——iOS开发之理解iOS中的MVC设计模式
模型-视图-控制器(Model-View-Controller,MVC)是Xerox PARC在20世纪80年代为编程语言Smalltalk-80发明的一种软件设计模式,至今已广泛应用于用户交互应用程 ...
- iOS开发之理解iOS中的MVC设计模式
模型-视图-控制器(Model-View-Controller,MVC)是Xerox PARC在20世纪80年代为编程语言Smalltalk-80发明的一种软件设计模式,至今已广泛应用于用户交互应用程 ...
- Objective-C——判断对象等同性
无论我们使用什么语言,总是会出现需要判断两个对象是否相等的情况,OC当然也不例外.首先看一段代码: NSString *str1 = [[NSString alloc] initWithCString ...
- iOS 将对象的属性和属性值拆分成key、value,通过字符串key来获取该属性的值
这篇博客光看标题或许就会产生疑问,某个对象,只要它存在某个属性,且值不是空的,不就能直接用点方法获取吗,为什么要拆分成key和value多此一举呢?下面,我用一个例子告诉大家,既然这方法是存在的,那就 ...
- 完全理解 Python 迭代对象、迭代器、生成器(转)
完全理解 Python 迭代对象.迭代器.生成器 本文源自RQ作者的一篇博文,原文是Iterables vs. Iterators vs. Generators » nvie.com,俺写的这篇文章是 ...
- 完全理解 Python 迭代对象、迭代器、生成器
完全理解 Python 迭代对象.迭代器.生成器 2017/05/29 · 基础知识 · 9 评论 · 可迭代对象, 生成器, 迭代器 分享到: 原文出处: liuzhijun 本文源自RQ作者 ...
- 深入理解Javascript window对象
首先看我们的源代码. <!DOCTYPE html> <html> <head> <meta charset="utf-8" /> ...
- 通过实现一个TableView来理解iOS UI编程
推荐一篇神作: 通过实现一个TableView来理解iOS UI编程 http://blog.jobbole.com/61101/
随机推荐
- 数据结构(C语言版)顺序表相关算法代码实现
这两天实现了一下顺序表的相关操作,包括顺序表初始化.创建.遍历.第i个元素前插入,删除第i个元素.查找元素e的位置.清空顺序表.销毁顺序表.合并两个非递减顺序表操作. 这次在网上学习到了新的布局方法, ...
- Java基础语法<二> 字符串String
1. 代码点与代码单元 Java字符串由char序列组成.大多数的常用Unicode字符使用一个代码单元就可以表示,而辅助字符需要一对代码单元表示. length()方法将返回采用UTF-16编码表示 ...
- Gulp安装流程、使用方法及cmd常用命令导览
Gulp安装流程.使用方法及CMD常用命令导览 来自前端小白的gulp及周边知识学习总结 一.名词介绍: Npm--node包管理工具 一开始我不理解,包管理工具是什么鬼.后来用到的gulp也好,gu ...
- JavaScript图片翻转
<script type="text/javascript"> /** * 注册函数f,当文档加载问成时执行这个函数f * 如果文件已经载入完成,尽快以异步方式执行它 ...
- smarty模板基本语法
smarty基本语法: 1.注释:<{* this is a comment *}>,注意左右分隔符的写法,要和自己定义的一致. <{* I am a Smarty comment, ...
- 关于github 0.6.2版本的使用方法
貌似做为一名前端开发人员,没听过使用过github,node,vue就像落伍一样,本人也是在前端自摸自爬的路上越走越远了,经常在群里听大神们讨论vue,github,node,好生羡慕,没人教,没人带 ...
- java多线程系列(三)---等待通知机制
等待通知机制 前言:本系列将从零开始讲解java多线程相关的技术,内容参考于<java多线程核心技术>与<java并发编程实战>等相关资料,希望站在巨人的肩膀上,再通过我的理解 ...
- Experience of Python Learning Week 1
1.The founder of python is Guido van Rossum ,he created it on Christmas in 1989, smriti of ABC langu ...
- aspcms多图调用以及错误提示:3704
1.“为师资介绍”(相册列表)建立了内容页(相册内容页), 需要对模板页面改造,在相册详细页调用多图,之前没有试过,这次利用: 实现多图调用,注意不能使用contentid=[content:id] ...
- spring mvc4使用及json 日期转换解决方案
spring mvc使用注解方式配制,以及对rest风格的支持,真是完美致极.下面将这两天研究到的问题做个总结,供参考.1.request对象的获取方式1:在controller方法上加入reques ...