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()是否使用正确的函数?
谢谢
您实际上并没有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)