Как кэшировать UIViewControllers для использования UICollectionViewCell? - PullRequest
0 голосов
/ 29 апреля 2019

Я создал UICollectionView с ячейками, содержащими несколько типов UIViewController представлений в качестве их contentView. Я хотел бы кэшировать UIViewController, которые не используются. Я попытался создать словарь используемых в настоящее время UIViewController на основе их indexPath, и набор неиспользованных UIViewController для использования UIViewController. Однако я заметил, что мои UIViewControllers не деинициализированы. Я основал свою реализацию на этих ответах Добавьте UIViewController в UICollectionViewCell и Заставьте PageViewController повторно использовать viewControllers, например tableView .

Ниже моя реализация

    func collectionView(_ collectionView: UICollectionView, willDisplay cell: UICollectionViewCell, forItemAt indexPath: IndexPath) {
        addViewController(toCell: cell, at: indexPath, of: viewTypes[indexPath.item])
    }

    func collectionView(_ collectionView: UICollectionView, didEndDisplaying cell: UICollectionViewCell, forItemAt indexPath: IndexPath) {
        removeViewController(fromCell: cell, at: indexPath)
     //   cleanCachedViewControllers(index: indexPath.item)
    }

    func addViewController(toCell cell:UICollectionViewCell, at indexPath:IndexPath,of type:ViewControllerType){
        var childViewController:UIViewController!

        if let cvc = currentViewControllers[indexPath]{
            childViewController = cvc
        }else{
            childViewController = TestViewController()
            currentViewControllers[indexPath] = childViewController
        }
        self.addChild(childViewController)
        childViewController.view.frame = cell.contentView.bounds
        cell.contentView.addSubview(childViewController.view)
        childViewController.didMove(toParent: self)
    }

    func removeViewController(fromCell cell:UICollectionViewCell, at indexPath:IndexPath){
        guard let childViewController = currentViewControllers[indexPath] else {
            return
        }
        print("remove view controller called")
        childViewController.willMove(toParent: nil)
        childViewController.view.removeFromSuperview()
        childViewController.removeFromParent()
    //        currentViewControllers[indexPath] = nil
   //         currentViewControllers.removeValue(forKey: indexPath)
        childViewController.view = nil
        recycledViewControllers.insert(childViewController)
    } 

    func getRecycledViewController(type:ViewControllerType)->UIViewController{

        var unusedViewControllers = recycledViewControllers.filter { (viewController) -> Bool in
            return viewController.isKind(of: type.type) && viewController.parent == nil
        }

        if let reusableViewController = unusedViewControllers.first{
            recycledViewControllers.remove(reusableViewController)
            return reusableViewController
        }else{
            let newViewController = type.initializeViewController()
            //            recycledViewControllers.append(newViewController)
            return newViewController
        }
    }

1 Ответ

0 голосов
/ 29 апреля 2019

Возможно, у вас есть циклическая ссылка, которая препятствует освобождению ваших объектов.

Xcode 8 и выше позволяет вам отлаживать и визуализировать текущий график памяти.Это очень полезно для выяснения причин, по которым объект не освобождается.

https://developer.apple.com/videos/play/wwdc2018/416/

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...