书籍链接:《Pro Swift》 (链接需要梯子才能打得开)。
布尔类型(Bool)、整型(Integer)、字符串(String)、多元组(Tuple)、枚举(Enum)、数组(Array)、字典(Dictionary)、结构(Struct)都是值类型。
class和closure都是引用类型。
一、Closures
举个例子,我们创建一个函数,并返回closure:
func createIncrementer() -> () -> Void {
var counter = 0
return {
counter += 1
print(counter)
}
}
在createIncrementer()
里面,有一个counter
变量,因为这个变量包含在返回的closure里面,所以他将会被捕获。运行下面代码后,会打印1和2:
let incrementer = createIncrementer()
incrementer()
incrementer()
因为closure是引用类型,所以运行下面代码之后,会打印1、2、3和4:
let incrementer = createIncrementer()
incrementer()
incrementer()
let incrementerCopy = incrementer
incrementerCopy()
incrementer()
二、struct和class
1. struct和class的对比
Swift的类和结构有很多共同点:
- 可以定义属性
- 可以定义方法
- 可以定义下标
- 可以定义初始化函数
- 可以使用扩展
- 可以遵循协议
类拥有而结构没有的功能:
- 继承
- 类型转换
- 类的实例可以使用Deinitializers清除他自己定义的资源
- 类实例可以被一个或多个地方引用
2. 使用struct和class的注意事项
- 如果不需要继承的话,尽量选择struct。
- 如果一定要选择class,把它声明为
final
(除非class需要被继承),这样可以提升性能。 - 不管是struct还是class,尽可能把属性定义为常量。
3. 不可变性
class和struct在不可变性层面是有区别的。
定义结构类型的Person:
struct PersonStruct {
var name: String
var age: Int
}
var taylor = PersonStruct(name: "Taylor Swift", age: 26)
taylor.name = "Justin Bieber"
上面的代码声明taylor
的时候,使用了var
,这意味着后面可以修改taylor
,但这明显是不合理的,一个Person被首次确定之后,不应该可以被修改的。如果我们改为let
:
let taylor = PersonStruct(name: "Taylor Swift", age: 26)
taylor.name = "Justin Bieber" // 这行代码会报错,因为`taylor`是常量,不能在修改
定义class类型的Person:
final class PersonClass {
var name: String
var age: Int
init(name: String, age: Int) {
self.name = name
self.age = age
} }
let taylor = PersonClass(name: "Taylor Swift", age: 26)
taylor.name = "Justin Bieber"
这里的代码编译通过。声明taylor
的时候,使用了let
,但是后面还可以修改name
,因为class是引用类型,这里的let
是指taylor
指向的PersonClass
实例的地址不变,当我们修改name
的时候,PersonClass
实例的地址是不变。
下面是其他例子:
// struct变量: 可以更改属性和实例
var taylor1 = PersonStruct(name: "Taylor Swift", age: 26)
taylor1.name = "Justin Bieber"
taylor1 = PersonStruct(name: "Justin Bieber", age: 22)
// struct常量: 不可以更改属性和实例
let taylor2 = PersonStruct(name: "Taylor Swift", age: 26)
//taylor2.name = "Justin Bieber"
//taylor2 = PersonStruct(name: "Justin Bieber", age: 22)
// class变量: 可以更改属性和引用
var taylor3 = PersonClass(name: "Taylor Swift", age: 26)
taylor3.name = "Justin Bieber"
taylor3 = PersonClass(name: "Justin Bieber", age: 22)
// class常量: 可以更改属性,但不能更改引用
let taylor4 = PersonClass(name: "Taylor Swift", age: 26)
taylor4.name = "Justin Bieber"
//taylor4 = PersonClass(name: "Justin Bieber", age: 22)
完
有任何问题,欢迎大家留言!
欢迎加入我管理的Swift开发群:536353151
,本群只讨论Swift相关内容。
原创文章,转载请注明出处。谢谢!