在RxSwift / RxCocoa中,您可以为委托创建一个反应式包装器(例如UIScrollViewDelegate
或CLLocationManagerDelegate
),以便为某些委托方法启用Rx可观察序列。
我正在尝试为UIApplicationDelegate
方法applicationDidBecomeActive:
到目前为止我尝试的内容非常简单,类似于RxCocoa中包含的DelegateProxy
子类。
我创建了我的DelegateProxy
子类:
class RxUIApplicationDelegateProxy: DelegateProxy, UIApplicationDelegate, DelegateProxyType {
static func currentDelegateFor(object: AnyObject) -> AnyObject? {
let application: UIApplication = object as! UIApplication
return application.delegate
}
static func setCurrentDelegate(delegate: AnyObject?, toObject object: AnyObject) {
let application: UIApplication = object as! UIApplication
application.delegate = delegate as? UIApplicationDelegate
}
}
UIApplication
的Rx扩展名:
extension UIApplication {
public var rx_delegate: DelegateProxy {
return proxyForObject(RxUIApplicationDelegateProxy.self, self)
}
public var rx_applicationDidBecomeActive: Observable<Void> {
return rx_delegate.observe("applicationDidBecomeActive:")
.map { _ in
return
}
}
}
在我的AppDelegate中,我订阅了observable:
func application(application: UIApplication, didFinishLaunchingWithOptions launchOptions: [NSObject: AnyObject]?) -> Bool {
// the usual setup
// and then:
application.rx_applicationDidBecomeActive
.subscribeNext { _ in
print("Active!")
}
.addDisposableTo(disposeBag)
return true
}
当我启动我的应用程序时,#34; Active!&#34;得到印刷然后我在RxCocoa的_RXDelegateProxy_
课程中遇到以下崩溃:
有人知道问题可能是什么吗?或者是否有人成功实施了rx_applicationDidBecomeActive
?
答案 0 :(得分:8)
对于RxSwift和内存管理来说,这看起来非常棘手。
RxUIApplicationDelegateProxy
的默认实现将委托代理的实例(在本例中为delegate
)设置为UIApplication
的{{1}}。
它还将原始AppDelegate
存储为名为forwardToDelegate
的属性,因此仍然可以将所有委托方法传递给它。
问题是,当设置新的app delegate时:
application.delegate = delegate as? UIApplicationDelegate
原来的一个被解除分配!您可以通过覆盖deinit
中的AppDelegate
进行检查。原因在this answer中解释。由于属性forwardToDelegate
的类型为assign
,因此当您的应用属性指向已解除分配的对象时,您的应用会崩溃。
我找到了解决方法。我不确定这是否是推荐的方式,所以请注意。您可以在DelegateProxyType
中覆盖RxUIApplicationDelegateProxy
的方法:
override func setForwardToDelegate(delegate: AnyObject?, retainDelegate: Bool) {
super.setForwardToDelegate(delegate, retainDelegate: true)
}
在正常情况下,您不希望保留代理,因为它会导致保留周期。但在这种特殊情况下,这不是问题 - 您的UIApplication
对象将存在,直到您的应用程序仍处于活动状态。