objective-c - 如何将事件对象添加到 NSMutableArray 并在它们被释放时删除它们?

标签 objective-c cocoa nsmutablearray

我有 Item 类和 List 类(它有一个 NSMutableArray)。

每次类 Item 被实例化(和销毁)时,它都会发布一个通知,该通知由类 List 监听。当类 List 收到通知时,会将类 Item 的实例添加到其列表中。

我正在尝试让 Item 类也发布一个通知,说明它即将被解除分配。问题是类 List 的 NSMutableArray 保留了类 Item 的实例。

处理这种情况的最合适的方法是什么?如果我在将它添加到 List 的数组时递减计数,那么当类 List 尝试调用 removeObject 时将抛出异常(因为它会尝试释放对象。)

基本上,我想要一个“监视器”类列表,其中包含 Item 的所有“事件”实例的列表。但是,我还需要能够释放/释放实例并让它们报告它们正在被释放,以便 List 可以从其 NSMutableArray 中删除它们。

谢谢你的帮助。

最佳答案

如果我理解正确,您想要一个维护对其项的弱引用而不是强引用的数组?

我不知道有什么方法可以用 cocoa 中的任何“内置”来做到这一点。我知道这样做的唯一方法是自己制作阵列,并将存储设为 __weak id[] .当对象解除分配时,这将自动将数组中的位置清零。如果你在保留释放模型下,你可以使用类似 MAZeroingWeakRef 的东西获得相同的行为。

这绝对是一个有趣的问题,我不知道有更简单的答案。我很想被证明是错的!

哈,我喜欢犯错!

有一个类叫 NSPointerArray 看起来它可以做你正在寻找的东西。但是,它仅在 Mac 上可用,并且仅在您使用垃圾收集时自动归零。

我会继续思考这个问题。这是一个有趣的问题! :)

所以我一直在思考这个问题,并想出了一个解决方案。它使用了两个非常规的东西:

  • NSMutableArray的子类(egads!)
  • 使用关联对象来确定对象释放

  • 首先,我必须对 NSMutableArray 进行子类化这样我就可以将一些自定义逻辑注入(inject) addObject: (和相关方法)。我不想通过 swizzling 来做到这一点,因为 NSArray和 friend 是一个类集群,混入/出集群充满危险。所以,一个子类。这很好,但我们将失去一些从“纯”中获得的很棒的功能 NSArray实例,例如 how they do weird things when they get big .哦,这就是生活。

    至于第二位,我需要一种让任意对象通知它即将或刚刚完成解除分配的方法。我想到了动态子类化对象的类,注入(inject)我自己的 dealloc/finalize方法,调用 super ,然后砸isa对象,但这似乎有点太疯狂了。

    所以,我决定利用一个叫做关联对象的有趣小东西。这些对于 ivars 来说就像类别对于类:它们允许您在运行时动态添加和删除伪实例变量。它们还具有通过对象解除分配自动清理的令人敬畏的副作用。所以我所做的只是创建了一个小的丢弃对象,当它被释放时发布一个通知,然后将它附加到常规对象。这样,当常规对象被释放时,丢弃的对象也将被释放,导致发布通知,然后我在 NSMutableArray 中监听该通知。子类。通知包含一个指向正在被销毁的对象的(陈旧)指针,但由于我只关心指针而不关心对象,所以没关系。

    所有这一切的结果是你可以做到:
    DDAutozeroingArray *array = [DDAutozeroingArray array];
    
    NSObject *o = [[NSObject alloc] init];
    [array addObject:o];
    NSLog(@"%ld", [array count]); //logs "1"
    [o release];
    NSLog(@"%ld", [array count]); //logs "0"
    

    源代码在 github 上,它(理论上)在 iOS 上应该和 Mac OS X 一样好(不管 GC 模式如何):https://github.com/davedelong/Demos

    干杯!

    ...我只是想到了一种没有自定义子类的方法,但我很累,明天会发布更新的答案。

    第二天早上……

    我刚刚用 NSMutableArray 更新了 Github 上的项目类别,允许您创建真正的NSMutableArray当它们被释放时自动归零它的对象。诀窍是创建一个 CFMutableArrayRef带定制retain设置正确观察的回调,然后只是转换 CFMutableArrayRefNSMutableArray并使用它(啊,免费桥接的魔力)。

    这意味着您现在可以执行以下操作:
    NSMutableArray *array = [NSMutableArray autozeroingArray];
    

    我加了一个 typedef将这些定义为 NSAutozeroingMutableArray ,只是为了明确说明,虽然这是一个 NSMutableArray ,它没有 retain它的对象就像一个普通的 NSMutableArray .然而,由于它只是一个 typedef而不是子类,您可以互换使用它们。

    关于objective-c - 如何将事件对象添加到 NSMutableArray 并在它们被释放时删除它们?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/5972808/

    相关文章:

    objective-c - objective-c 中多线程的正确方法?

    ios - 程序化语言本地化,无需重启

    objective-c - NSDocument 来保存一个完整的文件夹?

    objective-c - 上下文 NSMenu : how to get the NSTableView to which belongs to?

    iphone - 我的 Objective C 代码中只需要一个版本吗?

    ios - 在NSMutableArray中删除

    objective-c - 如何从 NSMutableArray 创建 NSMutableDictionary?

    objective-c - 将信息从 A 类委托(delegate)给 B 类再委托(delegate)给 C 类而不是直接从 A 类委托(delegate)给 C 是不是糟糕的编程设计?

    iphone - 按字符串值对字符串的 NSMutableArray 进行排序

    objective-c - 类似 nswindow 级别的 growl 通知