Swift真机调试问题-iOS 8.0中 xib描述的控制器view的子控件是nil的解决方案

最近在一个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,就会先去找控制器类名去掉Controllernib文件,如果找不到会再找控制器类名相同的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的原因

*以上纯属个人见解,如有说的不对请大神多多指教*

你可能感兴趣的:(Swift真机调试问题-iOS 8.0中 xib描述的控制器view的子控件是nil的解决方案)