RxSwift` .addDisposableTo(disposeBag)`导致内存泄漏

时间:2017-07-28 08:31:23

标签: ios swift memory-leaks instruments rx-swift

我在我的公司项目RxSwift中大量使用。当在仪器中运行性能测试时,出现了令人担忧的问题。

每次调用.addDisposableTo(disposeBag)时,Instruments都会显示大约10个字节的内存泄漏。没有具体的模式说明为什么会发生这种情况,比如在正确的地方不使用[weak self],它只是没有明显的原因发生。

一些示例代码:

class ContactsViewModel: NSObject {
    fileprivate let disposeBag = DisposeBag()
    fileprivate let provider = AuthorizedNetworking().provider

    var contacts: Variable<[User]> = Variable([])
    var suggestedContacts: Variable<[User]> = Variable([])

    func fetchContact(suggestions: Bool = false) {
        ActivityIndicator.showLoadingHUD(message: "Fetching contacts...")
        let observable = provider.request(suggestions ?
            .suggest :
            .searchContacts(query: nil, global: false)).filterSuccessfulStatusCodes()
        let mapped = observable.checkForErrors().mapObject(DataListResponse<User>.self)
        mapped.subscribe { [weak self] event in
            switch event {
            case let .next(response):
                ActivityIndicator.hideLoadingHUD()
                if response.success, let contacts = response.data {
                    if suggestions {
                        self?.suggestedContacts.value = contacts
                    } else {
                        self?.contacts.value = contacts.filter { $0.contactType == "Friend" }
                    }
                } else {
                    Log(.Network, .Error, "Unable to retrieve current user")
                }
            case let .error(error):
                ActivityIndicator.hideLoadingHUD()
                Log(.Network, .Error, error.localizedDescription)
            default:
                break
            }
            }.addDisposableTo(disposeBag) <- Instruments show leak [6 bytes] at this line

    }
}

我已经完成了一些研究,我有一个版本,仪器可能无法理解RxSwift并使其看起来有泄漏,但实际上,没有。 但是很可能我的实现存在一些我不知道的问题,因为我对RxSwift缺乏经验。 任何帮助表示赞赏。

1 个答案:

答案 0 :(得分:1)

我会使用RxSwift资源调试功能获得第二意见。您可以使用RxSwift.Resources.total变量调试内存泄漏,以确保问题是RxSwift,而不是仪器误报。

按照RxSwift问题#378中的说明启用调试模式,将此代码添加到您的应用代理中:

/* add somewhere in
func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplicationLaunchOptionsKey : Any]? = nil)
*/
_ = Observable<Int>.interval(1, scheduler: MainScheduler.instance)
    .subscribe(onNext: { _ in
        print("Resource count \(RxSwift.Resources.total)")
    })

并多次使用您的observable来查看在流程结束时资源总数是否始终相同。

最后,考虑到如果包含observable或dispose包的对象泄露,问题是容器对象,而不是observables。由于Facebook SDK的问题,我正在泄漏视图和控制器,我在调试控制器内部使用的observable时发现了问题。

我希望它有所帮助, 哈维