Effective Objective-C 2.0 编写高质量iOS与OS X代码 对象等同性

1. 若想检测对象的等同性,请提供“isEqual”与“hash”方法。
- (BOOL)isEqual:(id)object
{
     if(self == object) return YES;
     if([self class] != [object class] ) return NO;
     ConcreteId *otherId = (ConcreteId*)object;
     if(![someProperty isEqual:otherId.someProperty])
          return NO;
     // ……… 比较其他属性值
     return YES;
}

① 特定类所具有的等同性判定方法
如果受测的参数与接收该消息的对象都属于同一类,那么就调用自己编写的判定方法。否则使用超类来判断。
- (BOOL)isEqual:(id)object
{
     if([self class] ==[ object class ])
     {     
          return [self isEqualToPerson:(EOCPerson*)object};
     }else {
          return [self isEqual:object];
     }
}

2. 相同的对象必须具备相同的哈希码,但是两个哈希码相同的对象却未必相同。

3. 不要盲目地逐个检测每条属性,而是应该依照具体需求来制定检测方案。

① 等同性判定的执行深度。
如果某个对象是从数据库创建出来的,其中就有可能有个属性是“唯一标识符”,这种情况下只需使用这个属性进行判断而无需全部属性都判断一遍。

4. 编写hash时,应该使用计算速度快而且哈希码碰撞几率低的算法。


5. 容器中可变类的等同性。
① 在容器中放入可变类对象的时候,就不应该再修改其哈希码了。
如果放入collection后其哈希码又变化了,那么其所在的这个“箱子数组”对它来说就是“错误”的。

使用NSMutableSet 和 NSMutableArray 对象测试一下。先把一个数组加入set中:
NSMutableSet *set = [NSMutableSet new];
NSMutableArray *arrayA = [@[@1, @2] mutableCopy];
[set addObject:arrayA];
NSLog(@“set = %@“,set);
// set = {((1,2))}

再向set加入一个数组,此数组与前一个数组所含的对象相同,顺序也相同:
NSMutableArray *arrayB = [@[@1, @2] mutableCopy];
[set addObject:arrayB];
NSLog(@“set = %@“,set);
// set = {((1,2))} 

此时仍只有一个对象,因为刚才要加入的那个数组对象和set中已有的数组对象相等,所以set并不会改变:
NSMutableArray *arrayC = [@[@1] mutableCopy];
[set addObject:arrayC];
NSLog(@“set = %@“,set);
// set = {((1),(1,2))}  

然后我们再改变arrayC的内容:
[arrayC addObject:@2];
NSLog(@“set = %@“,set);
// set = {((1,2),(1,2))}  

此时set中就包含了两个彼此相等的数组,根据set语义是不允许出现这种情况的。然后现在却无法保证这一点。
若是拷贝此set
NSSet *setB = [set copy];
NSLog(@“setB = %@“,setB);
// setB = {((1,2))} 

所以说,如果把某个对象放入set后又修改其内容,那么后面的行为将很难预测。


摘取自:《Effective Objective-C 2.0 编写高质量iOS与OS X代码的52个有效方法》,详细请购买书籍,支持作者及译者。


Effective Objective-C 2.0 编写高质量iOS与OS X代码 对象等同性,,5-wow.com

郑重声明:本站内容如果来自互联网及其他传播媒体,其版权均属原媒体及文章作者所有。转载目的在于传递更多信息及用于网络分享,并不代表本站赞同其观点和对其真实性负责,也不构成任何其他建议。