10 结构体和类 - —— 《Swift3.0 从入门到出家》
结构体和类
Swift中的面向对象5个要素:枚举、结构体、类、协议、扩展
面向对象研究的是对象,完成一件事情需要多个对象参与,是生活的映射
Swift中结构体和类非常相似,也就是结构体能完成类的所有功能。结构体是值类型,类是引用类型
结构体
结构体定义格式:
struct 结构体名称{
成员变量(数据类型定义的变量)—属性
成员方法(函数) —行为
}
结构体的成员变量不要求初始化
如果结构体的成员变量赋初值,编译器会自动生成一个不带参数的构造器,原有的之一构造器还可以继续使用
结构体自动提供了逐一构造器init,对对象或者实例的所有属性赋值
结构体中的成员方法不能直接修改成员变量的值,如果修改成员变量的值,需要使用关键字mutating修饰该成员方法
重点:
通过实例或者对象调用的方法称为实例方法
类方法只能用类型名称(结构体类型名/类名)调用
static或者class修饰的函数,称其为类方法,class修饰函数只能类中使用
结构体实例方法可以直接访问结构体的成员变量
结构体的类方法默认不能访问结构体中的成员变量
实例方法可以直接调用其他实例方法,调用类方法可以直接使用类名调用
类方法中可以直接调用其他类方法,不能直接调用实例方法
init构造器
如果自定义的构造方法和系统提供的构造方法同名,系统的构造方法被替代
构造器中用self关键字对成员变量进行初始化,谁调用了该方法,self就相当于谁
类
类的定义格式:
class 类名{
成员变量
成员方法
}
结构体和类的区别
- Swift中的类没有统一的父类
- 类中的成员变量必须初始化,可以直接在定义变量的时候赋值,也可以自定义构造 方法赋值,如果自定义了构造方法,无论和系统提供的构造方法是否相同,系统提供的构造方法就不能再使用了
- 类中实例方法可以直接修改类中的成员变量
- 类提供的构造方法是没有参数的
- 类是引用类型,而结构体是值类型
- 类可以被继承,结构体不能被继承
- 与值类型不同,引用类型在被赋予到一个变量、常量或者被传递到一个函数时,操作的并不是其拷贝
- 一个类可以被定义成多个常量,定义了常量后,其数值不会再发生改变,没一次创建一个常量,修改的是后台类的值
可失败构造器 --- 可以类中定义可以在结构体中定义
当使用构造器创建对象时 可以向构造器传递的形参无效 或者在构造器中使用函数外部的资源缺失 就会造成创建对象失败 如果创建对象失败 调用任何实例方法都会崩溃 为了解决崩溃问题 使用可失败构造器 将崩溃的结果变成nil
init? 创建的对象可能存在nil值 所以当对象创建成功 需要强制解析
init! 创建的对象相当于使用了隐式解析 使用隐式解析或者强制解析的前提条件 确保对象真实存在
可失败构造器 必须在满足某个条件的情况下 才调用return nil的语句 证明对象创建失败
【注意】定义的可失败的构造器 一定不能和非可失败的构造器参数名称相同 类型相同 参数个数相同
例子:
init?(name: String) {//可失败构造器,可能有这个属性,也可能没这个属性
if name.isEmpty {
return nil
}
self.name = name
self.age = 10
self.height = 1.0
}
func study() -> Void {
print("正在写代码")
}
}
调用
var stu1 = Student.init(姓名: "韩梅梅", 年龄: 10, 身高: 1.20)
stu1.study()
var stu2 = Student.init()
stu2.study()
var stu3 = Student.init(name: "")
print(stu3)
stu3?.study()
类和结构体的选择
在你的代码中,可以使用类和结构体来的自定义数据类型
结构体总是通过值传递 类实例总是通过引用传递,意味着两者适用不同的任务
按照通用的准则,当符合一条或者多条以下条件时,请考虑使用 构建结构体
<1>结构体的主要目的是用来盛装少量相关简单数据值
<2>有理由预计一个结构体实例在赋值或者传递时,封装的数据将会被拷贝而不是被引用
<3>任何在结构体中储存的值类型属性,也将会被拷贝,而不是被引用
<4>结构体不需要去继承另一个已存在类型的属性或行为
合适的结构体候选者包括:
<1>几何形状的大小,封装一个width属性和height属性,两者均为Double类型
<2>一定范围内的路径,封装一个start属性和length属性,两者均为Int类型
三维坐标系内一点,封装x,y和z的属性,三者均为Double类型
属性观察者
属性观察者是用来观察属性变化前和变化后的值
willSet【(newValue)】
willSet方法在被观察的属性将要发生变化时调用该方法 该方法默认携带一个参数newValue 表示属性的新值
didSet【(oldValue)】
** didSet**方法在被观察的属性已经变化之后调用该方法 这个方法也携带一个默认参数oldValue 表示属性的旧值
和OC语言中KVO原理相同
例子:
class Person {
//一个属性带有一个观察者
var age = 1 {
willSet {
//观察属性将要变化的值
print("预祝你\(newValue)岁生日快乐🎂")
}
didSet {
//观察属性变化以后的值
print("恭喜你从\(oldValue)岁到\(age)岁")
}
}
//改变age的值
func happyBirthday() -> Void {
age += 1
}
}
**调用: **
var xiaoPerson = Person.init()
for i in 1...6 {
xiaoPerson.happyBirthday()
}
<练习>游戏公司观察1到6月份游戏的下载量 如果当月游戏下载量超过10万 工资翻倍
struct gameDownLoadNum {
var downLoadNum = 0 {
willSet {
print("本月的游戏下载量将要到达\(newValue)万次")
}
didSet {
print("上个月的下载量为\(oldValue)万次 本月的下载量为\(downLoadNum)万次,请注意查看")
if downLoadNum >= 10 {
print("恭喜你当月工资翻倍")
}
else {
print("下载量未超过10万继续努力")
}
}
}
mutating func changeNum() -> Void {
downLoadNum = Int(arc4random()) % 20
}
}
调用
var num = gameDownLoadNum.init()
for i in 1...6 {
num.changeNum()
}