类和结构体的相同点
- 定义属性用于储存值
- 定义方法用于提供功能
- 定义下标用于通过下标语法访问值
- 定义初始化器用于生成初始化值
- 通过扩展以增加默认实现的功能
- 符合协议以对某类提供标准功能
类与结构体相比的特殊功能
- 继承允许一个类继承另一个类的特征
- 类型转换允许在运行时检查和解释一个类实例的类型
- 取消初始化器允许一个类实例释放任何其所被分配的资源
- 引用计数允许对一个类的多次引用
重点
- 我们简单了解了类和结构体之间的区别和联系后,我们还必须了解一种重要的一点。
在swift中结构体和枚举是值类型,而类是引用类型。值类型被赋予给一个变量,常数或者本身被传递给一个函数的时候,实际上操作的是其的拷贝。
引用传递和值传递
我们看一下下面这个例子
class Person {
var name: String
init(name: String) {
self.name = name
}
}
- 如何证明类是引用传递
var person1 = Person(name: "Bill")
var person2 = person1
print("第一次打印person1的名字\(person1.name)--person2:\(person2.name)")
person2.name = "jason"
print("第二次打印person1的名字\(person1.name)--person2:\(person2.name)")
- 将Person类换成结构体,在来试试看
struct Person {
var name: String
}
== 和 ===
“等价于(===)”表示两个类类型(class type)的常量或者变量引用同一个类实例。
“等于(==)”表示两个实例的值“相等”或“相同”,判定时要遵照类设计者定义定义的评判标准,因此相比于“相等”,这是一种更加合适的叫法。
因为类是引用类型,有可能有多个常量和变量在后台同时引用某一个类实例。(对于结构体和枚举来说,这并不成立。因为它们作值类型,在被赋予到常量,变量或者传递到函数时,总是会被拷贝。)
看完例子,在思考一下
class Person: NSCopying {
var name: String
required init(name: String) {
self.name = name
}
func copy(with zone: NSZone? = nil) -> Any {
return type(of: self).init(name: self.name)
}
}
var person1 = Person(name: "Bill")
var person2 = person1
print(person1 === person2)
var person3 = person2.copy() as! Person
print(person2 === person3)
集合(Collection)类型的赋值和拷贝行为
Swift 中Array和Dictionary类型均以结构体的形式实现。然而当数组被赋予一个常量或变量,或被传递给一个函数或方法时,其拷贝行为与字典和其它结构体有些许不同。
以下对数组和结构体的行为描述与对NSArray和NSDictionary的行为描述在本质上不同,后者是以类的形式实现,前者是以结构体的形式实现。NSArray和NSDictionary实例总是以对已有实例引用,而不是拷贝的方式被赋值和传递。
- 到底什么时候执行copy行为,是swift后台系统所决定的,简单的赋值不会发生copy行为,只有赋值的集合内容发生了改变,比如增加,删除,修改等行为发生时才会触发这个copy行为。
以下是对于数组,字典,字符串和其它值的拷贝的描述。 在你的代码中,拷贝好像是确实是在有拷贝行为的地方产生过。然而,在Swift 的后台中,只有确有必要,实际(actual)拷贝才会被执行。Swift 管理所有的值拷贝以确保性能最优化的性能,所以你也没有必要去避免赋值以保证最优性能。(实际赋值由系统管理优化)
字典类型的赋值和拷贝
var ages = ["Peter": 23, "Wei": 35, "Anish": 65, "Katya": 19]
var copiedAges = ages
copiedAges["Peter"] = 24
println(ages["Peter"])
// 输出 "23"
- 这个字典的键(keys)是字符串(String)类型,值(values)是整(Int)类型。这两种类型在Swift 中都是值类型(value types),所以当字典被拷贝时,两者都会被拷贝。
数组类型的赋值和拷贝
- OC中的数组
let classArray1 = NSMutableArray(arrayLiteral: 1, 3)
let classArray2 = classArray1
classArray2[0] = 2
print("classArray1:\(classArray1), classArray2:\(classArray2)")
- swift中的数组
let classArray1 = [1,3]
var classArray2 = classArray1
classArray2[0] = 2
print("classArray1:\(classArray1), classArray2:\(classArray2)")
使用copy复制数组
- 下面这段代码在swfit3之后就编译不过去了
var array1 = ["s","d"]
array1.copy()
- 我们看一下我们自定义的类和结构体存在数组中是如何进行copy行为的。
类的
class Person: NSObject {
var name: String
init(name: String) {
self.name = name
}
}
- 进行测试
let person1 = Person(name: "bill")
let person2 = Person(name: "jason")
let array1 = [person1, person2]
let array2 = array1
print(String(format: "array1的地址%p", array1))
print(String(format: "array2的地址%p", array2))
array2[0].name = "jack"
print(String(format: "array1的地址%p--- arrary1的元素%@,", array1, array1))
print(String(format: "array2的地址%p--- arrary2的元素%@", array2, array2))
- 打印结果
结构体
struct Person {
var name: String
}
- 测试结果
喜欢楼主的关注一下毕竟第一次写博客。欢迎加我微信一起探讨技术。
下节