最近在一个Swift项目中做真机调试时发现的一个问题,不知道大家有没有碰到过。
使用xib去描述控制器的view,并且xib的名称与控制器的类名是一致的,在OC的时候一直是这么干的,到了Swift也是同样做法, 程序在iOS 9.0+ 的真机或模拟器中运行,并没有任何的问题
但因为程序需要适配iOS 8.0,找来了一台iOS 8.1.3的真机来运行程序,那么问题来了,有很多界面一跳转就crush,报了一个经典的错误(解包了一个空值),并发现这是一个xib描述的控制器
分析:
- 0.创建测试项目进行模拟错误的场景
- 1.空值是谁。
- 2.为什么造成空值
- 3.估计是Swift2.1的兼容性问题
创建test项目,运行环境:iOS 8.1.3,编译环境:Xocde7.2 1.
第一个界面我们使用storyboard来描述,并给它拖一个label,在viewDidLoad的方法里面访问这个label并给它的属性赋值,运行程序时没有任何问题.
class ViewController: UIViewController { /// Storyboard拖线的Label @IBOutlet weak var label: UILabel! override func viewDidLoad() { super.viewDidLoad() // 通过Storyboard拖线的label是有值的 label.text = "123" } override func touchesBegan(touches: Set, withEvent event: UIEvent?) {
// 点击控制器view的时候modal出用 xib 描述的DemoViewController
let demoVc = DemoViewController()
presentViewController(demoVc, animated: true, completion: nil)
}
}
2.创建一个 DemoViewController 的控制器,并且使用xib来描述,首席向这个控制器拖一个slider,并在viewDidload的时候访问slider,给它的value赋值。当我点击第一个界面的控制器view的时候弹出DemoViewController。
class DemoViewController: UIViewController {
//MARK:- 属性列表
@IBOutlet weak var slider: UISlider!
//MARK:- 生命周期方法
// 控制器view加载完毕
override func viewDidLoad() {
super.viewDidLoad()
// iOS 8.0 这里的slider 没有值
slider.value = 0.5 /* 就在这一行报错
fatal error: unexpectedly found nil while unwrapping an Optional value
*/
}
}
结果:跳转到 DemoViewController 时,出现了与之前一样的错误
还是解包了一个空值
并且 self 的 slider 为nil
第一个问题已经知道了: slider是空的
发现控制器view加载完成后slider依然是空的。通过拖线获得的属性默认就是强制解包的,这个时候去访问slider 就相当与解包了一个空值。也就是说控制器在初始化的时候没有去加载nib文件。
// 重写initWithNibName 监听控制器初始化
override init(nibName nibNameOrNil: String?, bundle nibBundleOrNil: NSBundle?) {
super.init(nibName: nibNameOrNil, bundle: nibBundleOrNil)
print("DemoViewController----initWithNilName")
}
required init?(coder aDecoder: NSCoder) {
fatalError("init(coder:) has not been implemented")
}
为了验证这个说法,我重写initWithNibName
这个方法,监听控制器的初始化,随便打印一下,看它是否有调用这个方法,
它的确有调用这个方法,说明它是底层实现可能出现了bug,因为当一个控制器初始化的时候它底层会调用 initWithNibName
,而
initWithNibName
会去判断nibName这个参数是否为nil,如果为nil,就会先去找控制器类名去掉Controller
的nib
文件,如果找不到会再找控制器类名相同的nib
文件。很明显,nib
文件在iOS 8.1.3
的环境下nib
文件并没有加载成功
override init(nibName nibNameOrNil: String?, bundle nibBundleOrNil: NSBundle?) {
super.init(nibName: "\(DemoViewController.self)", bundle: nil)
/* iOS8.0的真机上 系统也会自动掉用initWithNibName这个方法,但是控制器的子控件是空的
这说明,iOS8.0 initWithNibName内部没有加载到与当前控制器相对应的nib文件
苹果可能在iOS9.0修复了这个bug,为了适配iOS8.0,最好重写initWithNibName,并将nilName传进去
*/
print("DemoViewController----initWithNilName")
}
required init?(coder aDecoder: NSCoder) {
fatalError("init(coder:) has not been implemented")
}
最后我尝试将nibName
传进去,然后运行程序,这个问题就完美解决了。
总结:当控制器的nib不能被加载出来时,解决方案有两种
1> 将xib改成storyboard来实现,不过这个方法比较麻烦,首先storyboard比较重量级,所有控件或事件都要重新拖线,而且创建的时候需要创建
UIStoryboard
对象,还需要强转, 非常麻烦。(不推荐)2> 重写
init(nibName: , bundle: )
方法,将控制器的类名传进去即可,这方法比较方便,加一句代码就ok了
由于Swift的代码非常简洁,用起来很爽。同时可能因为它的语法改动太过频繁,可能会导致一些版本兼容性的问题。这可能就是很多开发者还不愿意使用swift的原因
*以上纯属个人见解,如有说的不对请大神多多指教*