标签:
son‘ has no initializers
-> ‘Person‘ 类没有实例化器s
原因:如果一个类中定义了必选属性,必须通过构造函数为这些必选属性分配空间并且设置初始值
重写
父类的构造函数/// `重写`父类的构造函数
override init() {
}
提示错误 Property ‘self.name‘ not initialized at implicitly generated super.init call
-> 属性 ‘self.name‘ 没有在隐式生成的 super.init 调用前被初始化
super.init()
调用/// `重写`父类的构造函数
override init() {
super.init()
}
提示错误 Property ‘self.name‘ not initialized at super.init call
-> 属性 ‘self.name‘ 没有在 super.init 调用前被初始化
/// `重写`父类的构造函数
override init() {
name = "张三"
age = 18
super.init()
}
func
/// 学生类
class Student: Person {
/// 学号
var no: String
override init() {
no = "001"
super.init()
}
}
super.init()
,保持代码执行线索的可读性super.init()
必须放在本类属性初始化的后面,保证本类属性全部初始化完成Optional
属性Optional
class Person: NSObject {
/// 姓名
var name: String?
/// 年龄
var age: Int?
}
可选属性
不需要设置初始值,默认初始值都是 nil可选属性
是在设置数值的时候才分配空间的,是延迟分配空间的,更加符合移动开发中延迟创建的原则
/// `重载`构造函数
///
/// - parameter name: 姓名
/// - parameter age: 年龄
///
/// - returns: Person 对象
init(name: String, age: Int) {
self.name = name
self.age = age
super.init()
}
init()
,则系统不再提供默认的构造函数init()
无法完成分配空间和设置初始值的工作重写
父类的构造函数/// `重写`父类构造函数
///
/// - parameter name: 姓名
/// - parameter age: 年龄
///
/// - returns: Student 对象
override init(name: String, age: Int) {
no = "002"
super.init(name: name, age: age)
}
重载
构造函数/// `重载`构造函数
///
/// - parameter name: 姓名
/// - parameter age: 年龄
/// - parameter no: 学号
///
/// - returns: Student 对象
init(name: String, age: Int, no: String) {
self.no = no
super.init(name: name, age: age)
}
注意:如果是重载的构造函数,必须
super
以完成父类属性的初始化工作
重载
和重写
重载
,函数名相同,参数名/参数个数不同
构造函数
withXXX...
重写
,子类需要在父类拥有方法的基础上进行扩展,需要 override
关键字
/// `重写`构造函数
///
/// - parameter dict: 字典
///
/// - returns: Person 对象
init(dict: [String: AnyObject]) {
setValuesForKeysWithDictionary(dict)
}
原因:
运行时
,动态向对象发送 setValue:ForKey:
方法,为对象的属性设置数值实例化
添加 super.init()
同样会报错
原因:
必选属性
必须在调用父类构造函数之前完成初始化分配工作将必选参数修改为可选参数,调整后的代码如下:
/// 个人模型
class Person: NSObject {
/// 姓名
var name: String?
/// 年龄
var age: Int?
/// `重写`构造函数
///
/// - parameter dict: 字典
///
/// - returns: Person 对象
init(dict: [String: AnyObject]) {
super.init()
setValuesForKeysWithDictionary(dict)
}
}
运行测试,仍然会报错
错误信息:this class is not key value coding-compliant for the key age.
-> 这个类的键值 age 与 键值编码不兼容
可选
的概念/// 姓名
var name: String?
/// 年龄
var age: Int = 0
/// `重写`构造函数
///
/// - parameter dict: 字典
///
/// - returns: Person 对象
init(dict: [String: AnyObject]) {
super.init()
setValuesForKeysWithDictionary(dict)
}
提示:在定义类时,基本数据类型属性一定要设置初始值,否则无法正常使用 KVC 设置数值
init(dict: [String: AnyObject]) {
super.init()
setValuesForKeysWithDictionary(dict)
}
override func setValue(value: AnyObject?, forKey key: String) {
print("Key \(key) \(value)")
super.setValue(value, forKey: key)
}
// `NSObject` 默认在发现没有定义的键值时,会抛出 `NSUndefinedKeyException` 异常
override func setValue(value: AnyObject?, forUndefinedKey key: String) {
print("UndefinedKey \(key) \(value)")
}
setValuesForKeysWithDictionary
会按照字典中的 key
重复调用 setValue:forKey
函数forUndefinedKey
函数,程序会直接崩溃
NSUndefinedKeyException
异常forUndefinedKey
,会保证 setValuesForKeysWithDictionary
继续遍历后续的 key
forUndefinedKey
,子类可以不必再实现此函数/// 学生类
class Student: Person {
/// 学号
var no: String?
}
Designated
convenience
关键字修饰的构造方法就是便利构造函数nil
self.init()
重写
或者 super
/// `便利构造函数`
///
/// - parameter name: 姓名
/// - parameter age: 年龄
///
/// - returns: Person 对象,如果年龄过小或者过大,返回 nil
convenience init?(name: String, age: Int) {
if age < 20 || age > 100 {
return nil
}
self.init(dict: ["name": name, "age": age])
}
注意:在 Xcode 中,输入
self.init
时没有智能提示
/// 学生类
class Student: Person {
/// 学号
var no: String?
convenience init?(name: String, age: Int, no: String) {
self.init(name: name, age: age)
self.no = no
}
}
指定构造函数
或者用 self.
的方式调用父类的便利构造函数
nil
标签:
原文地址:http://www.cnblogs.com/-liangjingjing/p/5350370.html