原文地址为http://www.tuicool.com/articles/EveeEf
Hi, There。今天這個篇文章要來介紹 Swift 中 struct 和 class 有什麼不一樣的地方?首先要先和大家提到一個观念,Value Type 和 Reference Type 其中 struct 是 Value Type 而 class 是 Reference Type 所以這篇文章呈現的 struct 的行为也可以套用到所有的 value type 物件,相同地 class 的行為也可以套用到 reference type 的物件上。
注:Value Type是值类型,reference type 是引用类型
值类型的变量直接包含他们的数据,而引用类型的变量存储对他们的数据引用,因此后者称为对象,因此对一个变量操作可能影响另一个变量所引用的对象。对于值类型都有他们自己的数据副本,因此对一个变量操作不可能影响另一个变量
我們來建立一個自已的 struct 名称为 SRectangle 。程式碼如下呈現。
struct SRectangle {
var width = 200
}
這個 struct 有一個 property 名為 width 。
再來建立一個 class 名為 CRectangle 也有一個叫做 width 的 property 程式碼如下
class CRectangle {
var width = 200
}
准备好了,我們先來看第一個区别
var sRect = SRectangle()
// 或者
sRect = SRectangle(width:300)
sRect.width // 結果就是 300
var cRet = CRectangle()
// class 不能直接用 CRectangle(width:300) 必需要定义一個 constructor
cRect.width // 為 200
主要的差別就是 class 在產生物件時不能很自然把 property 放在 constructor 的参数里
var sRect = SRectangle()
// 或者
var sRect2 = sRect
sRect2.width // 目前值是 200,因為 sRect 直接 copy 一份完整記憶體給 sRect2
sRect2.width = 500
sRect.width // sRect.width 值不受 sRect2 影響還是 200
var cRect = CRectangle()
// 或者
var cRect2 = cRect
cRect2.width // 目前值是 200,因為 sRect 直接 copy 一份完整記憶體給 sRect2
cRect2.width = 500
cRect.width // cRect.width 也改變成了 500
以上是 value type 和 reference type 最大的不同,在每次做 assignment 的時候,value type 都会复制一份完整相同的內容給另一個变量,而 class 則是把記憶體的位置給变量。所以 reference type 大多會保有一份记忆體而重复利用。
在 Swift 這個語言的特色之一就是可变动內容的变量和不可變動变动的变量用 var 和 let 來區別。如果一開始用 let 來修飾变量就會造成 compiler 的錯誤。如下
var name = "Michael"
name.write(" Pan") // 這樣是正常
let name = "Michael"
name.write(" Pan") // 會造成 compile 錯誤
大至也遵循這個規則如下
let sRect = SRectangle()
sRect.width = 500 // 會造成錯誤
let cRect = CRectangle()
cRect.width = 500 // 預設可以直接更改雖然是 let 修飾的 cRect
let 這個效果無法延申至 class 。不过要提醒大家在 Swift 常用的 String, Array, Dictionary 都是 struct 所以 let 是会有效果的
我們先要為 struct 和 class 新增一個 method 名為 changeWidth() 而且我們在不改变原始的程式碼為前提下,新增一個 method。如下
struct SRectangle {
var width = 200
}
extension SRectangle {
mutating func changeWidth(width:Int){
self.width = width
}
}
class CRectangle {
var width = 0
}
extension CRectangle {
func changeWidth(width:Int){
self.width = width
}
}
這裡用了 extension 新增 class 裡的 method 而不需要改到原本的程式碼。 以。struct 和 class 的差別是 struct 的 function 要去改变 property 的值的時候需要加上 mutating 字樣,而 class 不用這樣做
5. Struct 和 Class 裡的 function 共同特色 - Implicit External Parameter Name
function 在定义时,有一種特別的写法叫做 external parameter name 舉例來說 一般写法
func setSize( width:Int, height:Int) {
println("width \(width), height \(height)")
}
setSize(50, 100) // 直接丟入參數
External Parameter Name
func setSize(width width:Int, height height:Int) {
println("width \(width), height \(height)")
}
setSize(width:50, height:100) // 必需在參數值前加上 external parameter name
另一個簡寫
func setSize(#width:Int, #height:Int) {
println("width \(width), height \(height)")
}
setSize(width:50, height:100) // 當 External name 和 internal name 一樣的時候可以在 internal name 前面加上 # 就可以少寫 external name
這樣的行為對於一般 function 並沒有強制一定要寫 external name 不過在 struct 和 class 裡是強制使用的。如下
struct SRectangle {
var width = 200
}
extension SRectangle {
mutating func changeWidth(width:Int){
self.width = width
}
func setName(name:String, width:Int) {
println("name \(name), width is \(width)")
}
}
var sRect = SRectangle()
sRect.setName("Michael", width: 500)
class CRectangle {
var width = 0
}
extension CRectangle {
func changeWidth(width:Int){
self.width = width
}
func setName(name:String, width:Int) {
println("name \(name), width is \(width)")
}
}
var cRect = CRectangle()
cRect.setName("Michael", width: 500)
关于 struct 或是 class 裡的 function 第二個參數開始都強制加入 external parameter name。在定義的時候不用特別寫,就會有了。
物件導向語言,令人覺得強大的能力莫過於繼承,而 struct 沒有 继承的功能,只有 class 有。如下
class CRectangle {
var width = 0
}
extension CRectangle {
func changeWidth(width:Int){
self.width = width
}
}
class CSquare : CRectangle { // :CRectangle 代表繼承的語法
// CSquare 這個 class 看起來什麼內容都沒有
}
var cSquare = CSquare()
cSquare.changeWidth(300) // 可以直接使用父類別的 function
cSquare.width // 也可以直接使用從父類別繼承下來的 property
以上是簡單常用到的行為裡 struct 和 class 常令使用者搞混的情況,希望對大家有幫助。 歡迎提供更多建議。