目录
一,属性
二,方法
三,下标
四,继承
一,属性
1,存储属性和计算属性
- 存储属性
1>它存储在实例的内存中
2>结构体、类可以定义存储属性
3>枚举不可以定义存储属性,因为枚举实例只存储成员值和关联值
- 计算属性
1>它的本质是函数,不占用实例的内存
2>枚举、结构体、类都可以定义计算属性
3>只读计算属性:只有get
,没有set
- 实例
2,延迟存储属性
- 在第一次用到属性时才会进行初始化
-
lazy
属性必须是var
,不能是let
,因为let
必须在实例初始化完成之前就有值 - 结构体实例必须是
var
才能访问lazy
属性,因为在初始化lazy
属性时需要改变结构体的内存
3,属性观察器
- 可以为
var
且非lazy
的存储属性添加属性观察器 - 属性观察器包括
willSet
和didSet
两个方法 - 在定义时给属性设值不会触发属性观察器
- 在初始化器中给属性设值不会触发属性观察器
4,inout
- 实例
- 分析
inout
的本质是地址传递
存储属性:将属性的内存地址传递给
inout
参数,在函数内部直接修改属性的值
带属性观察器的存储属性、计算属性
1>将属性的值赋值给一个临时变量
2>将临时变量的内存地址传递给inout
参数,在函数内部修改临时变量的值
3>将临时变量的值赋值给属性
- 汇编
update
函数
修改
width
属性的值
修改
side
属性的值
修改
girth
属性的值
5,类型属性
- 对比
实例属性:通过实例访问
1>存储实例属性:存储在实例中,每个实例都有一份
2>计算实例属性
类型属性:通过类型访问
1>存储类型属性:存储在全局区,整个程序只有一份(本质是全局变量)
2>计算类型属性
- 特点
1>用
static
定义类型属性,类还可以用class
来定义
2>必须给存储类型属性设置初始值,因为类型没有初始化器来初始化存储属性
3>枚举可以定义存储类型属性,因为它不需要存储在枚举实例中
4>存储类型属性默认就是lazy
,并且是线程安全的
- 单例模式
二,方法
1,介绍
- 枚举、结构体、类都可以定义方法
- 实例方法通过实例调用,类型方法通过类型调用
- 在实例方法中
self
代表实例,在类型方法中self
代表类型
2,mutating
- 默认情况下值类型的属性不能被自身的方法修改
- 在方法前加上
mutating
关键字才允许修改
3,@discardableResult
- 默认情况下方法调用后返回值未被使用会报警告
- 在方法前加上
@discardableResult
关键字可以消除该警告
4,赋值给变量
三,下标
1,介绍
- 使用
subscript
可以给枚举、结构体、类添加下标功能 -
subscript
的语法类似于计算属性,本质也是函数
2,结构体和类
- 结构体(值类型):不能直接修改成员的值
- 类(引用类型):可以直接修改成员的值
3,多参数
四,继承
1,介绍
- 枚举、结构体不支持继承,只有类支持继承
-
Swift
没有像OC
那样的规定:任何类最终都要继承自某个基类 - 子类可以重写父类的方法、下标、属性
2,重写方法、下标
- 实例方法、下标
- 类型方法、下标
3,重写属性
- 说明
1>子类可以将父类的属性重写为计算属性,不可以重写为存储属性
2>只能重写var
属性,不能重写let
属性
3>子类重写后属性的权限不能小于父类属性的权限(如果父类是只读的,那么子类可以是只读的、也可以是读写的;如果父类是读写的,那么子类必须是读写的)
- 实例属性
- 类型属性
4,属性观察器
- 实例属性
- 类型属性
5,final
- 被
final
修饰的类,禁止被继承 - 被
final
修饰的方法、下标、属性,禁止被重写
6,方法调用
- 结构体
- 类