我正在学习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
}
}