您的位置:首页 > 移动开发 > Objective-C

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

2014-03-27 11:07 507 查看
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个有效方法》,详细请购买书籍,支持作者及译者。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: 
相关文章推荐