如何在Swift中使用@IBOutlet属性正确实现延迟实例化

Sch*_*uce 4 cocoa-touch objective-c ios swift

我正在使用Big Nerd Ranch的最新iOS书籍学习iOS开发.我选择在Swift中实现他们的应用程序.在他们的一个应用程序中,他们在Objective C中有以下代码:

- (UIView *)headerView
{
    // If you have not loaded the header view yet...
    if (!_headerView) {

        // Load HeaderView.xib
        [[NSBundle mainBundle] loadNibNamed:@"HeaderView" owner:self options:nil]
    }

    return _headerView;
}
Run Code Online (Sandbox Code Playgroud)

Apple的Swift指南"@IBOutlet":

当您在Swift中声明一个出口时,编译器会自动将该类型转换为弱隐式展开的可选项,并为其指定初始值nil.实际上,编译器替换了@IBOutlet var name:使用@IBOutlet弱var name输入:Type!=没有.

正如在swift中的Lazy loading Properties中所指出的那样,有几个不同的选项.在这篇文章中他们都没有明确提到使用@IBOutlet进行延迟初始化,所以我最好尽力实现他们的建议,并想知道什么是最佳实践.

尝试#1(失败):遵循类似的模式,如AppDelegate.swift中的示例.这带来了问题"'IBOutlet'属性要求属性是可变的"

@IBOutlet var headerView : UIView  {
    // If the HeaderView has not been loaded yet...
    if !_headerView {
        // Load HeaderView.xib
        NSBundle.mainBundle().loadNibNamed("HeaderView", owner: self, options: nil)
    }
        return _headerView!
}
var _headerView : UIView? = nil
Run Code Online (Sandbox Code Playgroud)

尝试#2(失败):使用"@lazy"与"@IBOutlet"的任何变体都没有用,因为"@lazy"需要初始值设定项,但是如果使用了闭包,那么"@IBOutlet"具有相同的问题来自尝试#1

尝试#3(成功?):这是我能够让这个工作的唯一方法.我从一个有点不同的问题,Swift中的Lazy属性初始化中得到了这个想法.我对正在发生的事情的理解是headerView实际上被声明为"@IBOutlet weak var headerView:UIView!= nil",只会用我拥有的TableViewController子类初始化一次,并且初始化将是"懒惰的",因为它只发生当需要加载TableViewController时.

@IBOutlet var headerView : UIView

func loadHeaderView() {
    // If the HeaderView has not been loaded yet...
    if !headerView {
        // Load HeaderView.xib
        println("loaded HeaderView")
        NSBundle.mainBundle().loadNibNamed("HeaderView", owner: self, options: nil)
    }
}

override func viewDidLoad() {
    super.viewDidLoad()
    loadHeaderView()
    tableView.tableHeaderView = headerView
}
Run Code Online (Sandbox Code Playgroud)

那么,如何改进呢?

viewDidLoad()是否使用正确的函数?

谢谢

jrt*_*ton 6

一个懒散的插座是没有意义的 - 如果它是一个插座,它在装入笔尖时填充,而不是从代码中填充.如果您从代码加载它,它不需要是一个插座,所以你可以使用@lazy.


Nat*_*ook 5

您实际上并没有headerView为该代码提供闭包,您将其声明为只读计算属性.@IBOutlet属性需要是可变的,所以XIB/Storyboard可以发挥它的魔力,所以你需要用getter和setter来实现它,如下所示:

@IBOutlet var headerView : UIView {
get {
    if !_headerView {
        NSBundle.mainBundle().loadNibNamed("HeaderView", owner: self, options: nil)
    }
    return _headerView!
}
set {
    _headerView = newValue
}
}
var _headerView : UIView? = nil
Run Code Online (Sandbox Code Playgroud)