ios - UIViewController initWithNibNamed :bundle: initialized two objects?

标签 ios swift storyboard xib

我有一个UIViewController,我已经在 Storyboard中使用了一段时间,没有任何问题。随着我的应用程序的增长,并且我在越来越多的地方使用该 View Controller ,我意识到我可能应该使其更加便携,而不是从各处全面地进行这么多的转场。我以前做过这样的拆分,所以我在这里做了我认为合乎逻辑的事情。我选择了该 View Controller ,剪切它,然后粘贴到一个空的 .xib 文件中。将每次对 performSegueWithIdentifier 的调用更改为 init(nibName:bundle:)presentViewController 后,我崩溃了,意外发现一个对象为零在viewDidLoad()...

我在每次 init(...) 调用之后、呈现 View Controller 之前设置该对象的值。 nil 对象是从 viewDidLoad() 调用的。这是个问题。我刚刚设置了这个,现在它消失了?!

我重写了 init(...) 方法,发现 init(nibName:bundle:) 中的 self 没有与 相同的内存地址viewDidLoad() 中的 code>self。也很奇怪。

我重写了其他 init() 方法,发现在调用呈现我的 View 后,我的对象正在通过 init(coder:) 再次实例化!这里的 self 恰好是我的属性为零的地方的 self!

我认为 init(coder:) 被调用的唯一原因是我正在从 .xib 加载 View ,但我认为这是在 init(nibNamed:bundle:) 中处理?根据文档,如果我从 Storyboard 加载,并且不触及前者,我确实会调用 init(coder:)...它还说 Nib 是在查询 Controller 的 View 之前不会加载。如果我理解正确的话,在我呈现 View 之前,我的 View 不应该被查询。由于只有当我呈现它时才会发生崩溃,因此问题可能源于此。

我被困在这里了。在呈现之前,我仍然需要将上下文信息获取到 View Controller 。我什至尝试创建一个代理类来在演示之前进行实例化和属性设置,但我仍然无法动摇第二个实例!我从 init(nibName:bundle:) 获取一个,从 init(coder:) 获取另一个。两者都没有呈现,后者给了我一个零对象错误。任何帮助理解为什么会这样,以及我如何解决这个错误(功能?)的帮助将不胜感激。谢谢!


更新:

一时兴起,我决定将 View Controller 粘贴回 Storyboard,与主层次结构分开,并尝试通过其标识符实例化它。有效!不完全确定是怎么回事,但乔治做到了!现在我的问题是:为什么? .xibs 有什么可怕的邪恶和禁忌是 Xcode 和 iOS 不会告诉我的?我对这种行为一点也不感到困惑。我会继续尝试使用 .xib,哪怕只是为了防止 Xcode 对我大喊大叫关于入口点......

最佳答案

我不知道 Xcode 正在做什么,但这里有两个我编写的帮助器方法,可以轻松实例化任何 Storyboard VC - 您只需要 Storyboard 名称和 VC 标识符(可选,否则将初始化 VC)。通过将我的 VC 拆分成许多不同的 Storyboard,我避免了处理 xib,同时仍然保持简单。一个将其加载到您选择的导航 Controller 中,另一个则自行返回:

struct StoryboardHelper {
///instantiates a VC with (optional) identifier viewController from storyboardName, pushes it to hierarcy of navigationController, and runs setup block on it, animated specifies whether the push is animated
internal static func showStoryboard(storyboardName: String, viewController: String?, navigationController: UINavigationController, animated: Bool = true, setup: (UIViewController) -> () ){
    let storyboard = UIStoryboard(name: storyboardName, bundle: nil)
    let destinationVC = viewController != nil ? storyboard.instantiateViewControllerWithIdentifier(viewController!) : storyboard.instantiateInitialViewController()!
    setup(destinationVC)
    navigationController.pushViewController(destinationVC, animated: animated)
  }

///instantiates and returns a VC with (optional) identifier viewController from storyboardName
internal static func instantiateViewControllerFromStoryboard(storyboardName: String, viewController: String?) -> UIViewController{
    let storyboard = UIStoryboard(name: storyboardName, bundle: nil)
    return viewController != nil ? storyboard.instantiateViewControllerWithIdentifier(viewController!) : storyboard.instantiateInitialViewController()!
  }
}

关于ios - UIViewController initWithNibNamed :bundle: initialized two objects?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/37627341/

相关文章:

ios - 将 Firestore 时间戳转换为日期数据类型

ios - UIView 在 swift 中的指定初始化器

ios - 如何让tableview控件适合整个屏幕

ios - Swift 自定义单元格以适合屏幕宽度

ios - DetailView 中的 SwiftUI 核心数据绑定(bind)文本字段

ios - 如何轻松地将函数四舍五入到最接近的百分之一

iPhone - GCD 第一次不工作

ios - 如何仅在选择休息选项卡栏项目上显示选项卡栏项目的标题,在 Swift 中将没有标题

arrays - 为什么在 swift 中无法在数组中添加 json 键值

ios - 导航栏从我的 Storyboard 项目中消失了