首页 > 解决方案 > iOS swift:定义一个子 ViewController

问题描述

我在 ViewController 中有一个 UITabBar。选择标签栏项目我添加不同的子视图控制器。并删除其他人。

但我从来没有在控制台上看到 deint 日志,只有 viewWillDisappear..

我怎样才能确保删除子视图控制器以节省内存?

这就是我创建子控制器的方式:

private lazy var lotteryViewController: LotteryViewController = {
        // Load Storyboard
        let storyboard = UIStoryboard(name: "TrophyRoom", bundle: Bundle.main)

        // Instantiate View Controller
        var viewController = storyboard.instantiateViewController(withIdentifier: "LotteryViewController") as! LotteryViewController

        // Add View Controller as Child View Controller
        self.add(asChildViewController: viewController)

        return viewController
    }()

添加子控制器:

private func add(asChildViewController viewController: UIViewController) {
        // Add Child View Controller
        addChildViewController(viewController)

        // Add Child View as Subview
        containerView.addSubview(viewController.view)

        // Configure Child View
        viewController.view.frame = containerView.bounds
        viewController.view.autoresizingMask = [.flexibleWidth, .flexibleHeight]

        // Notify Child View Controller
        viewController.didMove(toParentViewController: self)
    }

从父 viewController 调用的 remove 方法:

private func remove(asChildViewController viewController: UIViewController) {
        // Notify Child View Controller
        viewController.willMove(toParentViewController: nil)

        // Remove Child View From Superview
        viewController.view.removeFromSuperview()

        // Notify Child View Controller
        viewController.removeFromParentViewController()
    }

子控制器的日志方法:

override func viewWillAppear(_ animated: Bool) {
        super.viewWillAppear(animated)

        print("Sessions View Controller Will Appear")
    }

    override func viewWillDisappear(_ animated: Bool) {
        super.viewWillDisappear(animated)

        print("Sessions View Controller Will Disappear")
    }

    deinit {
        NotificationCenter.default.removeObserver(self)
        // Log if a view controller is being deinited
        print("\nDeinit: \(self.description)\n")
    }

标签: iosswiftmemory-managementuiviewcontroller

解决方案


类实例内存管理通过保留引用的计数来工作。 deinit意思是“这个对象即将不复存在,因为不再有对它的保留引用”。

当您添加子视图控制器和删除子视图控制器时,您正在正确地进行父子“跳舞”。所以从这个角度来看一切都很好。当您调用 时viewController.removeFromParentViewController,对它的一个保留引用肯定会消失,即父子保留引用(childControllers已删除并释放子的数组)。

但这并不意味着所有保留引用都会消失。事实上,被称为的事实表明存在其他一些deinit保留参考。你现在要做的就是找出那个参考是什么。但事实上,您似乎已经向我们展示了它是什么:

private lazy var lotteryViewController: LotteryViewController = {
    let storyboard = UIStoryboard(name: "TrophyRoom", bundle: Bundle.main)
    var viewController = storyboard.instantiateViewController(withIdentifier: "LotteryViewController") as! LotteryViewController
    self.add(asChildViewController: viewController)
    return viewController
}()


lotteryViewController是对子视图控制器的强(保留)引用。因此,即使在您将其作为孩子移除后,它仍使该视图控制器保持活动状态。

因此,您的体系结构存在一个弱点(或至少是一个意想不到的结果)(我的意思是您对lazy var带有定义和调用初始化程序的实例属性的好奇使用)。制作子视图控制器的方法是实例化视图控制器,将其作为其子级交给父级,然后放开视图控制器,使对它的唯一引用是父级。(如果您需要对子项的后续引用,您始终可以通过 parent获取它。)您没有这样做,因此您不会得到deinit子项从父项中删除的时间。


推荐阅读