gpt4 book ai didi

ios - 接收 swift ; ViewController 从未取消初始化

转载 作者:行者123 更新时间:2023-11-28 05:38:48 25 4
gpt4 key购买 nike

我有一个由 ViewModel(称为 MainViewModel)支持的 ViewController(称为 MainViewController)。

ViewModel 有一个变量,用于定义哪个子 ViewController MainViewController 应该作为其子项呈现。

我的问题是,当一个 child 被另一个 child 取代时,它永远不会deinit

代码如下:

主视图 Controller :

class MainViewController: UIViewController {

var viewModel: MainViewModel!
private let disposeBag = DisposeBag()

override func viewDidLoad() {
super.viewDidLoad()

viewModel.viewController
.subscribe(onNext: { [weak self] vc in
self?.newVc(vc)
})
.disposed(by: disposeBag)
}

static func instantiate(viewModel: MainViewModel) -> MainViewController {
let vc = MainViewController()
vc.viewModel = viewModel
return vc
}

private func newVc(_ vc: UIViewController) {
remove(childViewController: children.first)
addFullScreen(childViewController: vc)
}
}

主视图模型:

class MainViewModel {

lazy var viewController: Observable<UIViewController> = {
return Observable.just(ColorViewController(.green))
.delay(RxTimeInterval.seconds(3), scheduler: MainScheduler.instance)
.startWith(ColorViewController(.yellow))
}()

}

您在 MainViewModelviewController 变量中看到,它首先发出黄色的 ColorViewController,3 秒后发出绿色的。ColorViewController 是 UIViewController 的基本子类,具有彩色 View ,并覆盖了 deinit 方法。当移除黄色的 ColorViewController 时,不会调用此方法。

谁持有黄色 ColorViewController 的引用,如何修复它?

红利代码:

extension UIViewController {

public func addFullScreen(childViewController child: UIViewController) {
guard child.parent == nil else { return }

addChild(child)
view.addSubview(child.view)

child.view.translatesAutoresizingMaskIntoConstraints = false
NSLayoutConstraint.activate([
view.leadingAnchor.constraint(equalTo: child.view.leadingAnchor),
view.trailingAnchor.constraint(equalTo: child.view.trailingAnchor),
view.topAnchor.constraint(equalTo: child.view.topAnchor),
view.bottomAnchor.constraint(equalTo: child.view.bottomAnchor)
])

child.didMove(toParent: self)
}

public func remove(childViewController child: UIViewController?) {
guard let child = child else { return }
guard child.parent != nil else { return }

child.willMove(toParent: nil)
child.view.removeFromSuperview()
child.removeFromParent()
}
}

更新:

所以我将 viewController 变量更改为:

lazy var viewController: Observable<UIViewController> = {
return Observable<Int>.interval(RxTimeInterval.seconds(3), scheduler: MainScheduler.instance)
.scan(0, accumulator: { (prev, next) -> Int in return prev + 1 })
.map { index -> UIViewController in
let modul = index % 3
print("Index: \(index): \(modul)")
switch modul {
case 0: return ColorViewController(.yellow, tag: "Yellow")
case 1: return ColorViewController(.blue, tag: "Blue")
case 2: return ColorViewController(.green, tag: "Green")
default: return ColorViewController(.red, tag: "Red")
}
}.startWith(ColorViewController(.cyan, tag: "Initial 1"),
ColorViewController(.purple, tag: "Initial 2"))
.take(10)
}()

现在我看到 .map 中生成的所有 ColorViewController 都按预期取消了初始化。但是传递给 .startWith 的两个永远不会被取消初始化,甚至在 .take(10) 导致 Observable 完成之后也不会。这对任何人都有意义吗?

最佳答案

打开可视化内存调试器,找出谁持有你想释放的 View Controller 。这是一篇关于它的文章:https://useyourloaf.com/blog/xcode-visual-memory-debugger/

还有一段来自 Apple 的视频:https://developer.apple.com/videos/play/wwdc2018/416

关于ios - 接收 swift ; ViewController 从未取消初始化,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/57674504/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com