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



我正在学习Big Nerd Ranch最新的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;
}

Apple关于"@IBOutlet"的Swift指南:

当你在Swift中声明outlet时,编译器会自动将该类型转换为弱隐式未包装的可选类型,并为其分配一个初始值nil。实际上,编译器将@IBOutlet var name: Type替换为@IBOutlet弱变量name: Type!= nil。

正如在swift中的延迟加载属性中指出的那样,有几个不同的选项。他们在那篇文章中都没有明确提到@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

尝试#2(失败):使用"@lazy"与"@IBOutlet"的任何变体都不起作用,因为"@lazy"需要初始化器,但如果使用闭包,那么"@IBOutlet"就会出现与尝试#1相同的问题

尝试#3(成功?):这是我能让它工作的唯一方法。我从一个有点不同的问题中得到了这个想法,Swift中的惰性属性初始化。我对正在发生的事情的理解是headerView实际上被声明为"@IBOutlet弱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
}

那么,如何改进呢?

viewDidLoad()是正确的功能使用?

谢谢

延迟加载的outlet没有意义——如果它是一个outlet,它是在加载nib时填充的,而不是从代码中填充的。如果你是从代码中加载它,它不需要是一个出口,所以你可以使用@lazy.

您实际上并没有使用该代码为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

以下作品…

@IBOutlet lazy var headerView : UIView? = {
    return NSBundle.mainBundle().loadNibNamed("HeaderView", owner: self, options: nil)[0] as? UIView
    }()

然后设置headerView

override func viewDidLoad() {
  super.viewDidLoad()
  tableView.tableHeaderView = headerView
}

你也可以使用didSet:

 @IBOutlet weak var profileImageView: UIImageView! {
        didSet {
            profileImageView.image = profileImage
            profileImageView.layer.masksToBounds = true
            profileImageView.layer.cornerRadius = 16
        }
    }

最新更新