快速阵列/集合内存管理



我花了一个多星期的时间在线阅读各种文章和 StackOverflow 关于 UIViewController 中 Swift 数组内存管理的问答,我的应用程序仍然崩溃。希望你能帮上忙。

我有一个父类控制器和多个从父类继承的子类(使用示例简化),如下所示。

class ParentViewController: UIViewController, UITableViewDelegate, UITableViewDataSource {
    lazy var results: [AnyObject] = [AnyObject]()  //can contain various classes, such as Product, User, etc.
    func tableView(tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
        return results.count
    }
        ...       
}

class SubViewController: ParentViewController {
   ... loads results Array via REST API, then then appends to array using 
   results.append(Product())
}

随机地,我会遇到这样的崩溃:

EXC_BAD_ACCESS KERN_INVALID_ADDRESS at 0x0000000a1abb4b17
Thread : Crashed: com.apple.main-thread
0  libobjc.A.dylib                0x000000019478fe10 cache_getImp + 16
1  libobjc.A.dylib                0x0000000194784a6c lookUpImpOrForward + 540
2  libobjc.A.dylib                0x000000019478fdb8 _objc_msgSend_uncached_impcache + 56
3  libswiftCore.dylib             0x00000001004d4e40 _ZN5swift12metadataimpl14ValueWitnessesINS0_17ObjCRetainableBoxEE18initializeWithCopyEPNS_11OpaqueValueES5_PKNS_8MetadataE + 24
4  libswiftCore.dylib             0x000000010036e3cc _TFSa6appendU__fRGSaQ__FQ_T_ + 92

根据我的理解,我相信这是因为results在调用results.append()时已经不在记忆中了。

在UIViewController中实例化数组/集合的最佳实践是什么,尤其是当有子类时?以下是我的一些想法,任何反馈将不胜感激。

1.在父类中使用lazy。这是基本情况,会导致崩溃

lazy var results: [AnyObject] = [AnyObject]()

2.在SubViewController初始化,这仍然会导致崩溃,另外,我应该使用 deinit() 来释放内存吗?

in ParentViewController
var results: [AnyObject]
in SubViewController
    required init(coder aDecoder: NSCoder) {
        super.init(coder: aDecoder)
        self.results = [Product]()
    }

3.在父类中results可选,这将要求父类中的所有方法检查可选。我应该在子类中还是在viewDidLoad()init初始化数组?另外,我应该deinit结果吗?在父类中为零?

in ParentViewController
var results: [AnyObject]?
deinit {
    results = nil
}
in SubViewController
    required init(coder aDecoder: NSCoder) {
        super.init(coder: aDecoder)
        self.results = [Product]()
    }

任何建议将不胜感激!

我不知道您的视图控制器的生命周期和关系是什么,但我可靠地执行此操作的一种方法是将"结果"数组放置到您的 AppDelegate。当您使用视图控制器时,这保证存在,并且可以从它们访问。

let appDelegate = UIApplication.sharedApplication().delegate as AppDelegate
appDelegate.results.append(...)

您是否在主线程中添加从其余 API 获取的结果?如果没有,请尝试以下操作:

dispatch_async(dispatch_get_main_queue(), {
   results.append(objectsFromApi)
})

最新更新