iOS 通过 NSHashTable 和 NSPointerArray 实现一对多代理

在 iOS 程序开发中,代理(协议)作为重要的数据传递手段频频被使用,但是代理有一个弊端,只能一对一传值,如果有一个事件需要众多类全部能够被通知到,比如网络发生变化,用户登录状态发生变化等,纵然可以用KVO和通知来实现这样的功能,但是如何使用代理来实现呢

可能很多人想到使用数组来存储代理

var delegates:[XXDelegate] = []

一行代码搞定,简单完美,但是如果真写出这样的代码,只能说你还需要加深对“内存管理”这个概念的理解,首先我们谈谈为什么上面这行代码不可以使用,不过在谈这个问题之前,我们可以聊聊另一个问题:为什么 iOS 中 IBOutlet 的属性可以是weak的?

当然很多人都知道,这是因为子View被加入了self.view.subviews 这个数组中,所以可以使用weak,而这也就成了问题的关键:数组会强引用加入其中的元素,这也是上面这行代码不可以使用的原因,那么,如果存在一个集合类型,他可以弱引用加入其中的元素,不就Ok了

Foundation 还真为我们提供了这样的结构,还提供了三个,分别是NSHashTable,NSMapTable,NSPointerArray,他们对应着我们常用的结构:Set,Dictionary,Array,但是它们与常用的这三种结构又有很大的不同。

对传统的集合类型,他们有如下缺点:

  1. 无法存储nil (Swift的Array类型可以存储nil)
  2. 无法对加入其中的元素进行弱引用

而Foundation 提供的这三个类型,则弥补了上面的缺点,使我们可以对其中的元素进行弱引用,达到一对多进行代理通知的目的。

其基本使用思路和上面提到的是类似的,但是具体代码上需要做些变化。

private var weakDelegates:NSHashTable = NSHashTable(options: NSPointerFunctions.Options.weakMemory)
//注意这里需要使用 weakMemory 选项,否则不具有弱引用效果

func addDelegate(delegate:ServerManagerDelegate){
        let object:AnyObject = delegate as AnyObject
        if !self.weakDelegates.contains(object) {
            self.weakDelegates.add(object)
        }
    }

    func removeDelegate(delegate:ServerManagerDelegate){
        let object:AnyObject = delegate as AnyObject
        if self.weakDelegates.contains(object) {
            self.weakDelegates.remove(object)
        }
    }

当需要对全部代理进行通知的时候,我们可以

for i in weakDelegates.allObjects {
            if let delegate:XXDelegate = i as? XXDelegate {
                delegate.foo()
            }
        }

如果需要按照加入代理的时间顺序来进行通知,可以使用 NSPointerArray 来实现,代码都是类似的

如果想要深入了解 NSHashTable,NSMapTable,NSPointerArray,那么你可以看看这篇文章:http://www.saitjr.com/ios/nspointerarray-nsmaptable-nshashtable.html

你可能感兴趣的:(iOS 通过 NSHashTable 和 NSPointerArray 实现一对多代理)