Как создать слабый делегат без вызова "Экземпляр будет немедленно освобожден, потому что свойство tableViewDelegate является" слабым ""

Я пытаюсь выделить источник данных моего tableView в отдельный объект делегата. Поскольку этот делегат должен получить доступ к табличному представлению в какой-то момент, мне нужна ссылка на делегирующий объект в делегате; и так как оба являются классами, мне нужно избегать сильных ссылочных циклов, делая делегат weak

Чтобы добиться этого, я попробовал следующий код.

class MyViewController: UIViewController {

    @IBOutlet weak var tableView: UITableView!
    weak var tableViewDelegate: UITableViewDataSource?

    override func viewDidLoad() {
        super.viewDidLoad()
        tableViewDelegate = TableViewDelegate() // throwing a warning
        tableView.dataSource = tableViewDelegate
    }
}

Когда я пытаюсь создать экземпляр делегата, XCode выдает предупреждение: "Экземпляр будет немедленно освобожден, потому что свойство tableViewDelegate является" слабым "".

Чтобы исправить это, я делаю следующее:

class MyViewController: UIViewController {

    @IBOutlet weak var tableView: UITableView!
    weak var tableViewDelegate: UITableViewDataSource?

    override func viewDidLoad() {
        super.viewDidLoad()
        let delegate = TableViewDelegate() // worried this creates a strong reference.
        self.tableViewDelegate = delegate
        tableView.dataSource = delegate
    }
}

Пожалуйста, подтвердите, верно ли следующее: инициализируя делегат в методе viewDidLoad(), я не рискую создать сильную ссылку, потому что переменная, которая содержит этот экземпляр, освобождается, как только мы покидаем область действия этого метода. Или, говоря иначе: единственное время, когда нам нужно беспокоиться о том, что переменная (которая указывает на класс) создает сильную ссылку, это если переменная инициализируется на уровне класса и, следовательно, будет жить так же долго, как и класс.,

Это верно?

1 ответ

Пожалуйста, подтвердите, верно ли следующее: инициализируя делегат в методе viewDidLoad(), я не рискую создать сильную ссылку, потому что переменная, которая содержит этот экземпляр, освобождается, как только мы покидаем область действия этого метода.

Правильный. Сильная ссылка исчезает, как только объем, в котором let объявлены выходы.

К сожалению, это означает, что ваш делегат все еще будет освобожден. Все, что вы сделали, это заставили замолчать предупреждение.

По сути, вам нужно иметь сильную ссылку на делегата где-нибудь, иначе он сразу исчезнет. Я чувствую, что вы должны сделать ссылку в MyViewController сильный. Цикла строгой ссылки не будет, если ваш делегат не содержит строгой ссылки на контроллер представления. Если вам нужна ссылка на MyViewController в делегате сделайте его слабым, т. е. контроллеру представления принадлежит делегат, а не делегату принадлежит контроллер представления.

Вот как я решил эту проблему:

    let dataSource = MyDataSource()

    lazy var viewModel : MyViewModel = {
        let viewModel = MyViewModel(dataSource: dataSource)
        return viewModel
    }()

а затем в viewDidLoad():

    tableView.delegate = self
    tableView.dataSource = dataSource

Вы можете увидеть полный демонстрационный проект здесь

Другие вопросы по тегам