Custom objects as NSDictionary keys

本文深入探讨Objective-C与Swift两种编程语言在iOS开发领域的应用与区别,包括其核心特性、优缺点及实际案例分析,为开发者提供选择语言的指导。

http://bynomial.com/blog/?p=73


This post describes what is needed for any class to work as a key of NSDictionary or NSMutableDictionary.

The only thing that is really required is that your class adopt the NSCopying protocol.  That’s because every key is copied before being added to the dictionary.  I mention why this design decision was probably made below.

The Tricky Part

However, in many cases you’ll also want to override the

isEqual:

and

hash:

methods as well, with these signatures:

- (BOOL)isEqual:(id)anObject;
- (NSUInteger)hash;

Why? Without these methods, your key objects are only considered equal when they are literally the same object at the same point in memory. For example, suppose you have an NSObject subclass with only one instance variable called

string

, and you do this:

MyObject *obj1 = [MyObject myObjectWithString:@"hello!"];
MyObject *obj2 = [MyObject myObjectWithString:@"hello!"];
[myDict setObject:@"one" forKey:obj1];
[myDict setObject:@"two" forKey:obj2];

You might expect that you now have one key in the dictionary with string value @”hello!” and value @”two”, but in reality you’ll have two keys and two values. The only difference in the keys is where they are in memory, but by default this is what defines the “isEqual” relationship among custom NSObject subclasses.

Defining isEquals and hashing

You can theoretically get away with not defining these methods since you inherit their default implementation from NSObject. However, their default implementation doesn’t know about any of your custom instance variables, so it’s really up to you to handle things if you want more sophisticated dictionary behavior.

The isEqual: method should return YES if and only if the two objects (the receiver and anObject) are of the same type, and represent the same object. It’s up to you to determine what counts as “the same object”.

The hash method needs to return an unsigned integer that is very likely to be different for different objects (in this case “different” means the opposite of “isEqual:” in the sense of your custom isEqual: method). It is absolutely required that whenever

isEqual:

returns YES for two objects, their

hash

integer is identical as well (otherwise dictionary behavior is undefined, and will very likely break).  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.

Suppose we have a sample NSObject subclass called TwoDice with only two integers, diceOne and diceTwo, as instance variables. (I know the singular of “dice” is “die” but I’m sticking with “dice” because my friends never say die for a single dice. You know what I mean.) Then we might define the methods as follows:

- (NSUInteger)hash {
  return diceOne + 7 * diceTwo;
}

- (BOOL)isEqual:(id)anObject {
  if (![anObject isKindOfClass:[TwoDice class]]) return NO;
  TwoDice *otherDice = (TwoDice *)anObject;
  // This code cares about the order of dice (maybe one is red and one is blue, to distinguish).
  return diceOne == otherDice.diceOne && diceTwo == otherDice.diceTwo;
}

You’ll typically want something like the factor of 7 here to help avoid hash collisions. For example, if I didn’t multiply diceTwo by seven and instead just added, then we would have the same hash value for the pairs (2, 5) and (4, 3). This will not break the dictionary – it can handle false hash collisions like this – but it will slow it down. By carefully choosing your hash method, you’ll keep your dictionary nice and fast, the way it was meant to be.

Why keys are copied (possible gotcha)

You might be wondering why NSDictionary doesn’t just retain the keys, and does the extra work of copying.

Here’s why: Suppose we do something like this -

NSMutableString *string = [NSMutableString stringWithString:@"hi"];
[aDict setObject:@"one" forKey:string];
[string setString:@"there"];
[aDict setObject:@"two" forKey:string];

If the keys were only retained then the dictionary would be in a very weird state right now. It would have two internal bins – one hashed from “hi” and the other from “there”, but both stored value objects (“one” and “two”) would have back-pointers to the single mutable string object with current value “there”. If we did a lookup on @”hi” we would not get a hit because no key would currently have that value. In fact, no lookup at this point could possibly return “one”, so that we have basically wasted memory. Copying keys avoids this problem by keeping the keys the same from that point on.

Keep this tricky spot in mind if you’re using a mutable object as a key!  It can still get you by allowing you to add an object as a key, then later have a failed lookup for the “same” key with a changed internal value.

Adopting NSCopying

You only need to implement one method for this protocol, and it’s implementation is basically always the same:

- (id)copyWithZone:(NSZone *)zone {
  MyObject *objectCopy = [[MyObject allocWithZone:zone] init];
  // Copy over all instance variables from self to objectCopy.
  // Use deep copies for all strong pointers, shallow copies for weak.
  return objectCopy;
}

References

Apple’s topics-for-cocoa notes on using NSDictionary and NSMutableDictionary (quote):

Important: Because the dictionary copies each key, keys must conform to the 

NSCopying

protocol). You should also bear this in mind when choosing what objects to use as keys. Although you can use any object that adopts the 

NSCopying

protocol, it is typically bad design to use large objects such as instances of 

NSImage

since this may incur performance penalties.

NSObject class reference, including descriptions of the hash and isEqual: methods:

Discussion from that page on hash:

If two objects are equal (as determined by the 

<a href="#//apple_ref/occ/intfm/NSObject/isEqual:">isEqual:</a>

method), they must have the same hash value. This last point is particularly important if you define 

hash

in a subclass and intend to put instances of that subclass into a collection.

If a mutable object is added to a collection that uses hash values to determine the object’s position in the collection, the value returned by the 

hash

method of the object must not change while the object is in the collection. Therefore, either the 

hash

method must not rely on any of the object’s internal state information or you must make sure the object’s internal state information does not change while the object is in the collection. Thus, for example, a mutable dictionary can be put in a hash table but you must not change it while it is in there. (Note that it can be difficult to know whether or not a given object is in a collection.)

Discussion from that page on isEqual:

This method defines what it means for instances to be equal. For example, a container object might define two containers as equal if their corresponding objects all respond 

YES

to an 

isEqual:

request. See the 

<a href="../../../Classes/NSData_Class/Reference/Reference.html#//apple_ref/occ/cl/NSData" target="_top">NSData</a>

<a href="../../../Classes/NSDictionary_Class/Reference/Reference.html#//apple_ref/occ/cl/NSDictionary" target="_top">NSDictionary</a>

<a href="../../../Classes/NSArray_Class/NSArray.html#//apple_ref/occ/cl/NSArray" target="_top">NSArray</a>

, and 

<a href="../../../Classes/NSString_Class/Reference/NSString.html#//apple_ref/occ/cl/NSString" target="_top">NSString</a>

class specifications for examples of the use of this method.

If two objects are equal, they must have the same hash value. This last point is particularly important if you define 

isEqual:

in a subclass and intend to put instances of that subclass into a collection. Make sure you also define 

<a href="#//apple_ref/occ/intfm/NSObject/hash">hash</a>

in your subclass.

代码下载地址: https://pan.quark.cn/s/bc087ffa872a "测控电路课后习题详解"文件.pdf是一份极具价值的学术资料,其中系统地阐述了测控电路的基础理论、系统构造、核心特性及其实际应用领域。 以下是对该文献的深入解读和系统梳理:1.1测控电路在测控系统中的核心功能测控电路在测控系统的整体架构中扮演着不可或缺的角色。 它承担着对传感器输出信号进行放大、滤除杂音、提取有效信息等关键任务,并且依据测量与控制的需求,执行必要的计算、处理与变换操作,最终输出能够驱动执行机构运作的指令信号。 测控电路作为测控系统中最具可塑性的部分,具备易于放大信号、转换模式、传输数据以及适应多样化应用场景的优势。 1.2决定测控电路精确度的关键要素影响测控电路精确度的核心要素包括:(1)噪声与干扰的存在;(2)失调现象与漂移效应,尤其是温度引起的漂移;(3)线性表现与保真度水平;(4)输入输出阻抗的特性影响。 在这些要素中,噪声干扰与失调漂移(含温度效应)是最为关键的因素,需要给予高度关注。 1.3测控电路的适应性表现测控电路在测控系统中展现出高度的适应性,具体表现在:* 具备选择特定信号、灵活实施各类转换以及进行信号处理与运算的能力* 实现模数转换与数模转换功能* 在直流与交流、电压与电流信号之间进行灵活转换* 在幅值、相位、频率与脉宽信号等不同参数间进行转换* 实现量程调整功能* 对信号实施多样化的处理与运算,如计算平均值、差值、峰值、绝对值,进行求导数、积分运算等,以及实现非线性环节的线性化处理、逻辑判断等操作1.4测量电路输入信号类型对电路结构设计的影响测量电路的输入信号类型对其电路结构设计产生显著影响。 依据传感器的类型差异,输入信号的形态也呈现多样性。 主要可分为...
(IHAOAVOABPvsAOBPvsAVOABPvsPSOBP)非洲秃鹫融合天鹰优化BP天鹰优化BP非州秃鹫BP粒子群(Matlab代码实现)内容概要:本文档主要围绕多种智能优化算法在不同工程领域的应用展开,重点介绍了非洲秃鹫优化算法(AVOA)、天鹰优化算法(AO)与BP神经网络的融合改进,并与其他经典算法如粒子群优化(PSO)进行对比分析。所有案例均提供Matlab代码实现,涵盖电力系统优化、路径规划、微电网调度、无人机控制、信号处理等多个方向,强调算法的科研复现能力与实际仿真价值。文档还展示了丰富的技术服务体系,涉及机器学习、深度学习、路径规划、通信与电力系统等多个前沿领域。; 适合人群:具备一定Matlab编程基础,从事科研工作或工程仿真的研究生、高校教师及企业研发人员,尤其适用于从事智能优化算法研究与应用的相关技术人员。; 使用场景及目标:①用于学术论文复现,特别是SCI/EI期刊中关于优化算法的实验部分;②为电力系统、无人机、储能调度等领域提供算法仿真技术支持;③帮助研究人员快速掌握多种智能算法的实现方法并进行性能对比。; 阅读建议:建议结合文档提供的网盘资源下载完整代码,按照目录顺序逐步学习,重点关注算法原理与Matlab实现的结合方式,同时可将文中案例作为自身课题的参考模板进行二次开发与创新。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值