iOS使用自定義對(duì)象作為NSDictionary的key探索

從不喜歡搞優(yōu)雅高調(diào)的官方開(kāi)場(chǎng)白,多數(shù)都是直接上代碼!
根據(jù)蘋果API的文檔來(lái)看:The key for value. The key is copied (using copyWithZone:; keys must conform to the NSCopying protocol). If aKey already exists in the dictionary, anObject takes its place.OK,直接看代碼:

@interface WKeyCustom : NSObject<NSCopying>
@property (nonatomic, strong) NSString *name;
@end
@implementation WKeyCustom
//這樣做之后,就可以當(dāng)作key傳到NSDictionary中去了
-(id)copyWithZone:(NSZone *)zone
{
    WKeyCustom *aCopy = [[WKeyCustom allocWithZone:zone] init];
    if(aCopy)
    {
        [aCopy setName:[self.name copyWithZone:zone]];
    }
    return aCopy;
}
@end
- (void)doHashTest
{
        WKeyCustom *keyCustomA = [[WKeyCustom alloc] init];
        keyCustomA.name = @"keyA";
        
        WKeyCustom *keyCustomB = [[WKeyCustom alloc] init];
        keyCustomB.name = @"keyA";
        
        NSMutableDictionary *mdict = [[NSMutableDictionary alloc] init];
        [mdict setObject:@"keyA" forKey:keyCustomA];
        [mdict setObject:@"keyB" forKey:keyCustomB];
        NSLog(@"%@", mdict);//這里輸出啥???--(1)
        
        NSString *str = [mdict objectForKey:keyCustomA];
        NSLog(@"%@", str);//這里輸出啥???--(2)
    }

答案:

(1)//難到不該是只有一個(gè)對(duì)象么?至少期望是只有一個(gè)對(duì)象的。
{
    "<WKeyCustom: 0x618000019860>" = keyB;
    "<WKeyCustom: 0x618000019a80>" = keyA;
}
(2)//只實(shí)現(xiàn)了copyWithZone,只能保證放的進(jìn)去,卻沒(méi)辦法取出來(lái)
(null)

--------華麗的分割線-----------

如果我給WKeyCustom類增加一個(gè)isEqual方法呢,會(huì)不會(huì)是我們預(yù)期的結(jié)果呢?

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

你才剛才的輸出結(jié)果是什么?
答案:

{
    "<WKeyCustom: 0x618000208b70>" = keyA;
    "<WKeyCustom: 0x618000208b90>" = (null);//這是什么鬼?
}
2017-03-03 13:00:47.416 WiOSDemo[63915:1211904] (null)

這個(gè)結(jié)果很出乎意料吧,難到你猜對(duì)是這樣了?好吧,其實(shí)mdict的內(nèi)容是不一定的,多次運(yùn)行試試,每次運(yùn)行的結(jié)果可能都不太一樣,至少在我們的電腦上是這樣的。其實(shí)這是一個(gè)錯(cuò)誤的設(shè)計(jì),這種做法的結(jié)果是undefined。

--------華麗的分割線-----------

我們?cè)俳o我們的WKeyCustom增加一個(gè)方法,讓它實(shí)現(xiàn)我們預(yù)期的效果:

-(NSUInteger)hash
{
    return self.name.hash;
}

結(jié)果:

2017-03-03 13:26:33.103 WiOSDemo[64063:1227637] {
    "<WKeyCustom: 0x608000016260>" = keyB;
}
2017-03-03 13:26:33.104 WiOSDemo[64063:1227637] keyB

總結(jié)

  • 只要key遵循NSCoping協(xié)議,它確實(shí)可以放到NSDictionary里面
  • 你不實(shí)現(xiàn)isEqual:和hash方法也不會(huì)出錯(cuò),因?yàn)镹SObject實(shí)現(xiàn)了
  • 當(dāng)且僅當(dāng)兩個(gè)對(duì)象類型相同且你認(rèn)為相等(這種相等由你決定)時(shí), isEqual:才返回YES
  • 如果isEqual:返回YES,那么hash也必須返回YES;On the other hand, it is ok to occasionally have different objects with the same hash value, although it’s better if you minimize how often this happens.

References

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • 轉(zhuǎn)至元數(shù)據(jù)結(jié)尾創(chuàng)建: 董瀟偉,最新修改于: 十二月 23, 2016 轉(zhuǎn)至元數(shù)據(jù)起始第一章:isa和Class一....
    40c0490e5268閱讀 2,041評(píng)論 0 9
  • 官方文檔點(diǎn)這里:Key-Value Coding Programming Guide、NSKey?Value?Co...
    阿斯蘭iOS閱讀 1,718評(píng)論 0 1
  • 一、Foundation framework中用于收集cocoa對(duì)象(NSObject對(duì)象)的三種集合分別是: N...
    LZM輪回閱讀 4,549評(píng)論 1 1
  • 本文Demo的完整工程代碼, 參考這里的EqualAndHashDemo 目錄為什么要有isEqual方法? 如何...
    _Weak閱讀 464評(píng)論 0 0
  • 朋友J給我抱怨,說(shuō)是自己得去看心理醫(yī)生了,我問(wèn)為什么。他說(shuō)他每天晚上都會(huì)忍不住想自己死了以后,死了世界上再也沒(méi)有他...
    喂喂胡咧咧閱讀 353評(píng)論 0 1

友情鏈接更多精彩內(nèi)容