gpt4 book ai didi

swift - 在委托(delegate)方法中引用时,弱 var 导出会丢失 (=nil)

转载 作者:行者123 更新时间:2023-11-30 10:44:50 25 4
gpt4 key购买 nike

我的类中有一个 UICollectionView 声明为@IBOutlet 弱变量artworkCollectionView:UICollectionView!

该类中有一个委托(delegate)方法,由另外两个 View Controller 调用,其中一个 VC 是弹出窗口,另一个是普通 VC。

委托(delegate)方法从数据库获取一些数据,然后更新闭包内调用的 Collection View :self.artworkCollectionView.reloadData()

当弹出的 VC 调用委托(delegate)方法时,一切都很好。但是,当普通 VC 在访问 self.artworkCollectionView.reloadData() 时调用委托(delegate)方法时,它会收到臭名昭著的 fatal error :隐式展开可选值时意外发现 nil。

我已经检查了对单元格重用标识符的所有引用,一切都是正确的。我怀疑由于 UICollectionView 被声明为弱var,当我从当前类转到弹出窗口然后弹出窗口调用委托(delegate)方法时,引用不会丢失,但是当我从当前类转到普通类时VC 然后普通的 VC 调用委托(delegate)方法,对我的弱变量的引用丢失了,因此它被“视为”nil。

@IBOutlet weak var artworkCollectionView: UICollectionView!

override func viewDidLoad() {
super.viewDidLoad()
// Set up
artworkCollectionView.dataSource = self
artworkCollectionView.delegate = self
artworkCollectionView.isUserInteractionEnabled = true
artworkCollectionView.allowsSelection = true
artworkCollectionView.register(UINib(nibName:
"MyCollectionViewCell", bundle: nil),
forCellWithReuseIdentifier: "cell")
}


// delegate method
func reloadCollections() {

retrieveAlbumRatings { (isAlbum) in
if isAlbum {

self.retrieveAlbumData(completion: { (isFinished) in

if isFinished {
// Reload collection views
self.artworkCollectionView.reloadData()

}
})
}
}
}

如果我是对的,我的问题是:我怎样才能给weak varartworkCollectionView:UICollectionView!一个强引用,这样它就不会在从当前类到普通 VC 并返回的流程中丢失?

编辑:这是我迄今为止尝试过的:

  1. 从 socket 声明中删除“weak”,使其成为:@IBOutlet varartworkCollectionView:UICollectionView!但我遇到了同样的错误

  2. 我通过重写performSegue将artworkCollectionView传递给普通VC,然后将其作为委托(delegate)方法的参数传递回来。这不会给我 fatal error ,但它也不会重新加载 UICollectionView,因为我认为无论如何对 UICollectionView 导出的弱引用都会丢失。

感谢您的帮助(免责声明:我对 Swift 还很陌生......)

最佳答案

Inside this class there is one delegate method called by two other View Controllers, one of these VC is a pop up, the other one is a normal VC.

The delegate method gets some data from the database and then updates the collection view calling inside a closure: self.artworkCollectionView.reloadData()

  1. 流程似乎是您有一个包含上述代码的 VC,该 VC 可以打开一个弹出窗口,或者只是对“普通 VC”执行标准推送转场。
  2. 您希望在弹出 VC 或普通 VC 中进行某些操作,加载一些数据,然后当用户被引导回原始 VC 时,UICollectionView 将使用该数据进行更新.

您的问题如下:

I passed artworkCollectionView to the normal VC via override performSegue and then passed it back as an argument of the delegate method. This does not give me the fatal error but also it does not reload the UICollectionView because I think that anyway the weak reference to the UICollectionView outlet is lost. You shouldn't be passing anything around like this in most cases unless you have a really good reason to do so (I don't see one).

您希望在这里进行关注点分离。你必须仔细考虑你想要在 VC 之间传递什么,以避免在它们之间产生奇怪的依赖关系。由于多种原因,我不会通过 socket ,第一个是,如果您决定更改它,您现在必须在多个 VC 中跟踪 socket 。第二个是,它需要太多的心理体操来跟踪导出的状态,因为它被到处传递。导出也仅保证在生命周期的某些阶段进行设置。例如,如果您从 prepareForSegue:sender: 中的 segue 检索目标 VC 并尝试引用当时的导出,它们都将为零,因为它们尚未设置。

这些都是为什么包含上述代码的 VC 应该是唯一一个(也是唯一一个)保持对 artworkCollectionView 中显示的内容和时间的控制的充分理由。这里的问题是你如何解决这个问题,而不是让弹出窗口或普通 VC 调用委托(delegate)方法或做一些奇怪的事情,例如将导出从一个 VC 传递到另一个 VC,而只是传递数据。

最简单的例子是:

  1. 弹出 VC 和普通 VC 调用一些代码来实际获取数据。
  2. 然后取决于您实际如何继续弹出 VC 或来自原始 VC 的普通 VC,使用 parentViewControllerpresentingViewController 以获取对原始 VC 的引用。
  3. 通过该引用将数据设置到原始 VC 中。
  4. 如有必要,关闭弹出 VC 或普通 VC(取决于您的特定应用,也许您希望用户按下 UIButton 来关闭,而不是为他们执行此操作)。
  5. 当原始 VC 重新出现时,向生命周期方法添加一些代码,例如viewWillAppear 将数据内容加载到UICollectionView 当时。

我认为您没有理由要传递原始 VC 之外的任何网点,而该 VC 应该是管理该网点的网点。

关于swift - 在委托(delegate)方法中引用时,弱 var 导出会丢失 (=nil),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/55959504/

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