類和結(jié)構(gòu)體的相同點(diǎn)
- 定義屬性用于儲存值
- 定義方法用于提供功能
- 定義下標(biāo)用于通過下標(biāo)語法訪問值
- 定義初始化器用于生成初始化值
- 通過擴(kuò)展以增加默認(rèn)實(shí)現(xiàn)的功能
- 符合協(xié)議以對某類提供標(biāo)準(zhǔn)功能
類與結(jié)構(gòu)體相比的特殊功能
- 繼承允許一個(gè)類繼承另一個(gè)類的特征
- 類型轉(zhuǎn)換允許在運(yùn)行時(shí)檢查和解釋一個(gè)類實(shí)例的類型
- 取消初始化器允許一個(gè)類實(shí)例釋放任何其所被分配的資源
- 引用計(jì)數(shù)允許對一個(gè)類的多次引用
重點(diǎn)
- 我們簡單了解了類和結(jié)構(gòu)體之間的區(qū)別和聯(lián)系后,我們還必須了解一種重要的一點(diǎn)。
在swift中結(jié)構(gòu)體和枚舉是值類型,而類是引用類型。值類型被賦予給一個(gè)變量,常數(shù)或者本身被傳遞給一個(gè)函數(shù)的時(shí)候,實(shí)際上操作的是其的拷貝。
引用傳遞和值傳遞
我們看一下下面這個(gè)例子
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類換成結(jié)構(gòu)體,在來試試看
struct Person {
var name: String
}
== 和 ===
“等價(jià)于(===)”表示兩個(gè)類類型(class type)的常量或者變量引用同一個(gè)類實(shí)例。
“等于(==)”表示兩個(gè)實(shí)例的值“相等”或“相同”,判定時(shí)要遵照類設(shè)計(jì)者定義定義的評判標(biāo)準(zhǔn),因此相比于“相等”,這是一種更加合適的叫法。
因?yàn)轭愂且妙愋停锌赡苡卸鄠€(gè)常量和變量在后臺同時(shí)引用某一個(gè)類實(shí)例。(對于結(jié)構(gòu)體和枚舉來說,這并不成立。因?yàn)樗鼈冏髦殿愋停诒毁x予到常量,變量或者傳遞到函數(shù)時(shí),總是會被拷貝。)
看完例子,在思考一下
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類型均以結(jié)構(gòu)體的形式實(shí)現(xiàn)。然而當(dāng)數(shù)組被賦予一個(gè)常量或變量,或被傳遞給一個(gè)函數(shù)或方法時(shí),其拷貝行為與字典和其它結(jié)構(gòu)體有些許不同。
以下對數(shù)組和結(jié)構(gòu)體的行為描述與對NSArray和NSDictionary的行為描述在本質(zhì)上不同,后者是以類的形式實(shí)現(xiàn),前者是以結(jié)構(gòu)體的形式實(shí)現(xiàn)。NSArray和NSDictionary實(shí)例總是以對已有實(shí)例引用,而不是拷貝的方式被賦值和傳遞。
- 到底什么時(shí)候執(zhí)行copy行為,是swift后臺系統(tǒng)所決定的,簡單的賦值不會發(fā)生copy行為,只有賦值的集合內(nèi)容發(fā)生了改變,比如增加,刪除,修改等行為發(fā)生時(shí)才會觸發(fā)這個(gè)copy行為。
以下是對于數(shù)組,字典,字符串和其它值的拷貝的描述。 在你的代碼中,拷貝好像是確實(shí)是在有拷貝行為的地方產(chǎn)生過。然而,在Swift 的后臺中,只有確有必要,實(shí)際(actual)拷貝才會被執(zhí)行。Swift 管理所有的值拷貝以確保性能最優(yōu)化的性能,所以你也沒有必要去避免賦值以保證最優(yōu)性能。(實(shí)際賦值由系統(tǒng)管理優(yōu)化)
字典類型的賦值和拷貝
var ages = ["Peter": 23, "Wei": 35, "Anish": 65, "Katya": 19]
var copiedAges = ages
copiedAges["Peter"] = 24
println(ages["Peter"])
// 輸出 "23"
- 這個(gè)字典的鍵(keys)是字符串(String)類型,值(values)是整(Int)類型。這兩種類型在Swift 中都是值類型(value types),所以當(dāng)字典被拷貝時(shí),兩者都會被拷貝。
數(shù)組類型的賦值和拷貝
- OC中的數(shù)組
let classArray1 = NSMutableArray(arrayLiteral: 1, 3)
let classArray2 = classArray1
classArray2[0] = 2
print("classArray1:\(classArray1), classArray2:\(classArray2)")
- swift中的數(shù)組
let classArray1 = [1,3]
var classArray2 = classArray1
classArray2[0] = 2
print("classArray1:\(classArray1), classArray2:\(classArray2)")
使用copy復(fù)制數(shù)組
- 下面這段代碼在swfit3之后就編譯不過去了
var array1 = ["s","d"]
array1.copy()
- 我們看一下我們自定義的類和結(jié)構(gòu)體存在數(shù)組中是如何進(jìn)行copy行為的。
類的
class Person: NSObject {
var name: String
init(name: String) {
self.name = name
}
}
- 進(jìn)行測試
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))
- 打印結(jié)果
結(jié)構(gòu)體
struct Person {
var name: String
}
- 測試結(jié)果
喜歡樓主的關(guān)注一下畢竟第一次寫博客。歡迎加我微信一起探討技術(shù)。
下節(jié)