Swift語(yǔ)言中碰凶,具有類(lèi)特征的類(lèi)型包括三種暮芭,即枚舉類(lèi)型、結(jié)構(gòu)類(lèi)型(包括基本類(lèi)型欲低,基本類(lèi)型實(shí)際都是結(jié)構(gòu)類(lèi)型的特例)辕宏、類(lèi)。其中枚舉類(lèi)型砾莱、結(jié)構(gòu)類(lèi)型是屬于值類(lèi)型瑞筐,類(lèi)屬于引用類(lèi)型。三種類(lèi)型都可以添加屬性腊瑟、方法聚假、下標(biāo)方法,能夠使用擴(kuò)展進(jìn)行功能擴(kuò)展闰非,使用協(xié)議等膘格。
枚舉
枚舉定義了一種包含一組相關(guān)值的公共類(lèi)型。枚舉是Swift中的一種與類(lèi)類(lèi)似的類(lèi)型财松,具有許多傳統(tǒng)類(lèi)才有的特征瘪贱,例如計(jì)算屬性纱控、實(shí)例方法,能夠通過(guò)擴(kuò)展或協(xié)議增強(qiáng)功能等菜秦。
- 枚舉定義甜害,Swift 語(yǔ)言的枚舉類(lèi)型的定義語(yǔ)法如下:
enum CompassPoint {
case North
case South
case East
case West
}
枚舉語(yǔ)法以一個(gè)關(guān)鍵字enum來(lái)標(biāo)識(shí),enum后面包含一個(gè)枚舉類(lèi)型名字球昨,枚舉定義全部放到一對(duì)大括號(hào)中尔店。 在枚舉中定義的值稱(chēng)為枚舉成員值,用case關(guān)鍵字來(lái)指示一個(gè)新的枚舉成員值褪尝。 與C和Objective-C語(yǔ)言的枚舉類(lèi)型不同的是:在Swift中不需要為枚舉成員分配一個(gè)默認(rèn)的整數(shù)值闹获。 如果為枚舉成員提供值,該值可以是一個(gè)字符串河哑、一個(gè)字符或者是一個(gè)任意整數(shù)或浮點(diǎn)數(shù)。枚舉成員值可以定義到一行中龟虎,并用逗號(hào)分割璃谨。
enum Planet {
case Mercury,Venus,Earth,Mars,Jupiter,Saturn,Uranus,Neptune
}
- 枚舉的使用
可以為一個(gè)常量或變量分配一種枚舉類(lèi)型的值,如:
var directionToHead =CompassPoint.West
以上定義的變量directionToHead可以推斷為是一種CompassPoint類(lèi)型的枚舉變量鲤妥,因此你可以設(shè)置該變量為CompassPoint類(lèi)型的其它值佳吞,如:
directionToHead = .East //枚舉類(lèi)型被省略
枚舉也能夠在switch語(yǔ)句中使用,用來(lái)匹配獨(dú)立的枚舉值:
directionToHead = .South
switch directionToHead {
case .North:
println("Lots of planets have a north")
case .South:
println("Watch out for penguins")
case .East:
println("Where the sun rises")
case .West:
println("Where the skies are blue")
}
- 為枚舉成員分配相關(guān)值
Swift中能夠?yàn)槊恳粋€(gè)枚舉成員規(guī)定一個(gè)任意類(lèi)型的相關(guān)值棉安,并且為每個(gè)枚舉成員規(guī)定的相關(guān)值的類(lèi)型可以不同底扳。
enum Barcode {
case UPCA(Int,Int,Int)
case QRCode(String)
}
該例子定義了一個(gè)類(lèi)型為Barcode的枚舉類(lèi)型,并定義了兩個(gè)枚舉值UPCA 和QRCode贡耽,并可以為枚舉值UPCA 分配一個(gè)多元組類(lèi)型的相關(guān)值衷模,為QRCode分配一個(gè)字符串類(lèi)型的相關(guān)值,該例子沒(méi)有為枚舉值本身指定任何類(lèi)型的值蒲赂。
可以使用以上定義的枚舉為一個(gè)常量或變量賦值阱冶,如:
var productBarcode =Barcode.UPCA(8,85909_51226,3)
還可以在switch語(yǔ)句中使用該枚舉,并通過(guò)綁定常量或變量的方式引出其帶有的相關(guān)類(lèi)型的值:
switch productBarcode {
case .UPCA(let numberSystem,let identifier,let check):
println("UPC-A with value of\(numberSystem),\(identifier),\(check).")
case .QRCode(let productCode):
println("QR code with value of\(productCode).")
}
如果枚舉成員的所有的相關(guān)值都作為常量被引出滥嘴,或者所有的相關(guān)值都作為變量形式被引出木蹬,以上語(yǔ)法還可以簡(jiǎn)寫(xiě)為如下形式:
switch productBarcode {
case let .UPCA(numberSystem,identifier,check):
println("UPC-A with value of\(numberSystem),\(identifier),\(check).")
case let .QRCode(productCode):
println("QR code with value of\(productCode).")
}
- 為枚舉分配原始值
除了為枚舉成員分配相關(guān)的值外,還能為每個(gè)枚舉成員預(yù)分配一個(gè)同類(lèi)型的原始值若皱。這與C 語(yǔ)言為枚舉成員分配一個(gè)整數(shù)值類(lèi)似镊叁,但Swift定義的原始值的類(lèi)型可以是字符串、字符走触、或任意的整數(shù)或浮點(diǎn)數(shù)類(lèi)型等晦譬,如:
enum ASCIIControlCharacter:Character {
case Tab ="\t"
case LineFeed ="\n"
case CarriageReturn ="\r"
}
該例子中,定義了一個(gè)含有三個(gè)枚舉成員的枚舉類(lèi)型ASCIIControlCharacter饺汹,并指定其原始類(lèi)型為字符類(lèi)型蛔添,并為每個(gè)枚舉成員分配一個(gè)字符類(lèi)型的默認(rèn)原始值。
與C語(yǔ)言為枚舉成員指定值類(lèi)似,Swift要求為枚舉的每個(gè)枚舉成員分配的原始值必須在枚舉聲明內(nèi)唯一迎瞧。當(dāng)使用整數(shù)類(lèi)型的原始值時(shí)夸溶,枚舉成員的其它原始值如果沒(méi)有指定,其能夠在第一個(gè)枚舉成員定義值的基礎(chǔ)上自動(dòng)加1凶硅,如下所示:
enum Planet:Int {
case Mercury =1,Venus,Earth,Mars,Jupiter,Saturn,Uranus,Neptune
}
為枚舉成員定義的原始值與為枚舉成員分配的相關(guān)值不同缝裁,枚舉成員的原始值是在枚舉第一次定義時(shí)被分配的,而枚舉成員的相關(guān)值雖然其類(lèi)型也是在枚舉定義時(shí)指定足绅,但其值是在使用枚舉類(lèi)型創(chuàng)建一個(gè)常量或變量時(shí)設(shè)置的捷绑。
在Swift中,能夠使用枚舉成員的rawValue方法來(lái)獲取枚舉成員的原始值:
let earthsOrder = Planet.Earth.rawValue()
結(jié)構(gòu)與類(lèi)
- 兩者比較
在Swift中氢妈,結(jié)構(gòu)和類(lèi)功能上幾乎相同粹污,兩者都具有如下相同的功能:- 可以定義屬性,用來(lái)存儲(chǔ)值首量;
- 可以定義方法壮吩,用來(lái)提供功能;
- 可以定義一個(gè)腳本方法加缘,用來(lái)使用腳本語(yǔ)法來(lái)存取它們的值鸭叙;
- 能夠定義初始化方法來(lái)設(shè)置它們的初始狀態(tài);
- 能夠擴(kuò)展來(lái)增加原先沒(méi)有實(shí)現(xiàn)的功能拣宏;
- 能夠遵從相關(guān)協(xié)議來(lái)提供確定類(lèi)型的標(biāo)準(zhǔn)功能沈贝。
類(lèi)在以下方面與結(jié)構(gòu)存在差別: - 類(lèi)能夠繼承,一個(gè)類(lèi)能夠繼承它的超類(lèi)的特性勋乾,而結(jié)構(gòu)不能繼承宋下;
- 類(lèi)允許在運(yùn)行時(shí)檢查和解釋一個(gè)類(lèi)實(shí)例的類(lèi)型;
- 類(lèi)可以帶有析構(gòu)函數(shù)市俊,允許類(lèi)的實(shí)例釋放它所分配的任何資源杨凑;
- 引用計(jì)數(shù)允許一個(gè)類(lèi)實(shí)例有多個(gè)引用。
- 結(jié)構(gòu)在代碼中總是以復(fù)制方式來(lái)傳遞摆昧,而不使用引用計(jì)數(shù)撩满。
- 類(lèi)和結(jié)構(gòu)的定義和實(shí)例化
類(lèi)和結(jié)構(gòu)的定義采用相似的語(yǔ)法,類(lèi)使用class關(guān)鍵字來(lái)指示绅你,結(jié)構(gòu)使用struct關(guān)鍵字來(lái)指示伺帘。
struct Resolution {
var width =0
var height =0
}
class VideoMode {
var resolution =Resolution()
var interlaced =false
var frameRate =0.0
var name:String?
}
每個(gè)新定義的類(lèi)或結(jié)構(gòu)都定義了一種新的類(lèi)型。
上面例子定義了一種稱(chēng)作Resolution的新的結(jié)構(gòu)類(lèi)型忌锯,其中包含和定義了兩個(gè)變量類(lèi)型的屬性伪嫁。還定義了一個(gè)稱(chēng)作VideoMode的新類(lèi),該類(lèi)定義和包含四個(gè)變量類(lèi)型的屬性偶垮,其第一個(gè)屬性resolution還使用了剛剛定義的結(jié)構(gòu)Resolution的實(shí)例進(jìn)行了初始化张咳。
類(lèi)和結(jié)構(gòu)中定義的變量或常量類(lèi)型的屬性像通常變量和常量一樣進(jìn)行初始化和賦值能真,屬性的類(lèi)型可以根據(jù)為其提供的初始值進(jìn)行推斷姚建。
為類(lèi)和結(jié)構(gòu)創(chuàng)建實(shí)例的語(yǔ)法相同:
let someResolution =Resolution()
let someVideoMode =VideoMode()
該例子采用了結(jié)構(gòu)和類(lèi)初始化最簡(jiǎn)單的語(yǔ)法形式(結(jié)構(gòu)和類(lèi)的類(lèi)型名面跟著一對(duì)圓括號(hào))。該初始化語(yǔ)法為結(jié)構(gòu)和類(lèi)創(chuàng)建了一個(gè)新的各自的實(shí)例,并賦值給兩個(gè)常量织鲸,兩個(gè)實(shí)例的屬性也在該初始化方法中被初始化為它們的默認(rèn)值缸濒。
在Swift中捉邢,所有的結(jié)構(gòu)類(lèi)型都會(huì)自動(dòng)產(chǎn)生一個(gè)參數(shù)初始化方法康震,可以使用該方法來(lái)初始化和創(chuàng)建結(jié)構(gòu)的新的實(shí)例及其成員屬性,新創(chuàng)建實(shí)例的屬性的初始值使用該初始化方法的參數(shù)傳進(jìn)來(lái)的值提鸟,如:
let vga =Resolution(width:640,height:480)军援。
而類(lèi)沒(méi)有提供相對(duì)應(yīng)的默認(rèn)的參數(shù)初始化方法。
在Swift中称勋,與腳本語(yǔ)言類(lèi)似胸哥,可以使用點(diǎn)語(yǔ)法的形式來(lái)存取一個(gè)結(jié)構(gòu)或類(lèi)實(shí)例的屬性以及子屬性,即讀取和設(shè)置其值:
讀取屬性的值:
println("The width of someResolution is\(someResolution.width)”)
println("The width of someVideoMode is\(someVideoMode.resolution.width)”)
設(shè)置一個(gè)屬性的值:
someVideoMode.resolution.width = 1280
- 結(jié)構(gòu)赡鲜、類(lèi)與枚舉的類(lèi)型
在Swift中烘嘱,結(jié)構(gòu)和枚舉與其它基本類(lèi)型(整型、浮點(diǎn)類(lèi)型蝗蛙、布爾類(lèi)型、字符串醉鳖、數(shù)組和詞典捡硅,這些類(lèi)型其實(shí)都是以結(jié)構(gòu)類(lèi)型實(shí)現(xiàn)的)一樣屬于值類(lèi)型。這意味著它們?cè)诜峙浣o一個(gè)變量或常量時(shí)或當(dāng)它作為參數(shù)傳送給一個(gè)函數(shù)時(shí)盗棵,它們的實(shí)例以及它們包含的所有作為值類(lèi)型的屬性一一被拷貝壮韭。
在Swift中,為了提供性能纹因,拷貝采用延遲拷貝的機(jī)制喷屋,即在實(shí)際用到時(shí)才拷貝。
let hd = Resolution(width:1920, height:1080)
var cinema =hd
在該例子中瞭恰,由于Resolution是一個(gè)結(jié)構(gòu)類(lèi)型屯曹,因此常量hd和變量cinema屬于Resolution結(jié)構(gòu)類(lèi)型的不同實(shí)例,因?yàn)樵跒樽兞縞inema賦值時(shí)發(fā)生了拷貝行為惊畏。
與結(jié)構(gòu)和枚舉不同恶耽,類(lèi)的類(lèi)型屬于引用類(lèi)型。引用類(lèi)型的實(shí)例在分配給一個(gè)變量或常量時(shí)或當(dāng)它作為參數(shù)傳送給一個(gè)函數(shù)時(shí)颜启,沒(méi)有拷貝發(fā)生偷俭。
let tenEighty =VideoMode()
tenEighty.frameRate = 25.0
let alsoTenEighty =tenEighty
alsoTenEighty.frameRate =30.0
由于VideoMode屬于類(lèi),因此現(xiàn)在兩個(gè)常量tenEighty和alsoTenEighty引用的是相同的VideoMode實(shí)例缰盏,只是對(duì)應(yīng)相同實(shí)例的不同的名字涌萤,因此這兩個(gè)常量的屬性值frameRate現(xiàn)在都等于30.0淹遵。
注意上面的tenEighty和alsoTenEighty被聲明為兩個(gè)常量,而不是變量负溪,這是因?yàn)閠enEighty和alsoTenEighty本身存儲(chǔ)的只是VideoMode的實(shí)例的引用值透揣,而不是VideoMode實(shí)例本身,因此你通過(guò)它們對(duì)引用的類(lèi)實(shí)例的屬性的改變笙以,改變是類(lèi)實(shí)例本身的屬性淌实,而不是引用本身。
由于類(lèi)是引用類(lèi)型猖腕,就如以上例子所示拆祈,多個(gè)變量或常量可能引用一個(gè)類(lèi)的相同的實(shí)例。為了判斷兩個(gè)常量或變量是否引用的是一個(gè)類(lèi)的相同實(shí)例倘感,Swift提供了兩個(gè)引用比較操作符: ‘===’與 ‘!==’放坏。可以使用這兩個(gè)操作符來(lái)檢查兩個(gè)常量或變量是否引用的是相同的實(shí)例:
if tenEighty ===alsoTenEighty {
println("tenEighty and alsoTenEighty refer to the same Resolution instance.")
}
在Swift中老玛,一個(gè)常量或變量引用一個(gè)類(lèi)的實(shí)例淤年,這與C語(yǔ)言中的指針類(lèi)似,但在Swift 中蜡豹,引用不是直接指向內(nèi)存中一個(gè)地址麸粮,因此不需要使用C語(yǔ)言中類(lèi)似的指針?lè)?hào)’*’,用來(lái)代表一個(gè)引用或指針镜廉。