iOS學(xué)習(xí)筆記44-Swift(四)枚舉和結(jié)構(gòu)體

一、Swift的枚舉

枚舉是一系相關(guān)聯(lián)的值定義的一個公共的組類型,同時能夠讓你在編程的時候在類型安全的情況下去使用這些值。
Swift中的枚舉比OC中的枚舉強大得多, 因為Swift中的枚舉是一等類型,它除了可以定義枚舉值外,還可以在枚舉中像類一樣定義屬性和方法

1. 簡單枚舉定義和使用
//定義枚舉,使用enum關(guān)鍵字
enum Method{
    case Add
    case Sub
    case Mul
    case Div
}
//可以連在一起寫,成員之間用“,“隔開
enum CompassPoint {
    case North, South, East, West
}
// 可以使用枚舉類型變量或常量接收枚舉值,枚舉值前有個點
var method: Method = .Add
// 注意: 如果變量或常量沒有指定類型, 那么前面必須加上該值屬于哪個枚舉類型
var point = CompassPoint.North
2. 枚舉和switch語句結(jié)合進行值匹配
method = Method.Sub
// 注意: 如果case中包含了所有的值, 可以不寫default
// 如果case中沒有包含枚舉中所有的值, 必須寫default
switch(method){
    case Method.Add:
        print("加法")
    case .Sub:// 如果變量已經(jīng)指定了枚舉類型,可以把前面的枚舉類型省略
        print("減法")
    case .Mul:
        print("除法")
    case .Div:
        print("乘法")
    default:
        print("都不是")
}
3. 枚舉的原始值

OC中枚舉的本質(zhì)就是整數(shù),所以OC中的枚舉是有原始值的,默認是從0開始,而Swift中的枚舉默認是沒有原始值的,但是可以在定義時告訴系統(tǒng)讓枚舉有原始值

枚舉定義原始值:
//定義枚舉類型為Int類型,默認從0開始,后面逐一加一
enum CompassPoint: Int {
    case North, South, East, West
}
//定義枚舉類型為Int類型,從指定值開始,后面逐一加一
enum Movement: Int {
    case Left = 5, Right, Top, Bottom
}
//除了Int類型,Swift枚舉更加強大,還可以定義為Double、String等
//但是如果指定除Int的其他類型,需要給所有枚舉值賦值
enum Method: String {
    case Add = "add"
    case Sub = "sub"
    case Mul = "mul"
    case Div = "div"
}
enum Constants: Double {
    case π = 3.14159
    case e = 2.71828
    case φ = 1.61803398874
    case λ = 1.30357
}
枚舉值和原始值之間的轉(zhuǎn)化:

// 獲取枚舉值對應(yīng)的原始值
println("Method.Add原始值為:\(Method.Add.rawValue)")
//打印:Method.Add原始值為:add

/*
通過原始值創(chuàng)建枚舉值
注意: 
1.原始值區(qū)分大小寫
2.返回的是一個可選類型值,因為原始值對應(yīng)的枚舉值不一定存在
*/
let method = Method(rawValue: "add")
// 由于返回是可選類型, 所以有可能為nil, 最好使用可選綁定
if let opE = Method(rawValue: "sub"){
    switch (opE){
        case .Add:
            print("加法")
        case .Sub:
            print("減法")
        case .Mul:
            print("除法")
        case .Div:
            print("乘法")
    }
}
4. 枚舉的關(guān)聯(lián)值

枚舉的關(guān)聯(lián)值是將額外信息附加到枚舉值中的一種極好的方式。使用關(guān)聯(lián)值,每一個枚舉值就可以是在某種模式下的一些特定值。
打個比方,你正在開發(fā)一款交易引擎,可能存在“買”和“賣”兩種不同的交易類型。除此之外每手交易還要制定明確的股票名稱和交易數(shù)量

枚舉的關(guān)聯(lián)值使用
//定義一個交易枚舉
enum TradeTmp {
    case Buy(String, Int) //買,關(guān)聯(lián)一個字符串和一個整形
    case Sell(String, Int) //賣,關(guān)聯(lián)一個字符串和一個整形
    case Borrow(String, Int, String) //借,每個枚舉值的關(guān)聯(lián)類型可以不一樣
}
//重新定義一個交易枚舉,為關(guān)聯(lián)值加上標(biāo)簽說明
enum Trade {
    case Buy(stock: String, amount: Int) //買,關(guān)聯(lián)股票名和交易數(shù)量
    case Sell(stock: String, amount: Int) //賣,關(guān)聯(lián)股票名和交易數(shù)量
}
//創(chuàng)建一個枚舉,關(guān)聯(lián)某些值
var tradeBuy = Trade.Buy(stock: "百度", amount: 2000)
var tradeBuy2 = Trade.Buy(stock: "APPL", amount: 4000)
var tradeSell = Trade.Sell(stock: "APPL", amount: 1000)
//第一種方式提取關(guān)聯(lián)值,利用switch語句提取關(guān)聯(lián)值
switch(tradeBuy){
    case .Buy(let stock, let amount):
        print("Buy \(stock) with \(amount) number")
    case let .Sell(stock, amount)://簡化
        print("Sell \(stock) with \(amount) number")
}
//第二種方式提取關(guān)聯(lián)值,使用模式匹配提取關(guān)聯(lián)值
if case let Trade.Sell(stock, amount) = tradeSell {
    print("Sell \(amount) of \(stock)")
}
5. 枚舉的屬性

盡管增加一個存儲屬性到枚舉中不被允許,但你依然能夠創(chuàng)建計算屬性。當(dāng)然,計算屬性的內(nèi)容都是建立在枚舉值下或者枚舉關(guān)聯(lián)值得到的。

//定義枚舉,添加一個計算屬性
enum Device {
    case iPad, iPhone
    var year: Int {
        switch self {
            case iPhone: return 2007
            case iPad: return 2010
        }
    }
}
//創(chuàng)建一個枚舉值
var device = Device.iPad
print("iPad is \(device.year)") //結(jié)果:iPad is 2010
6. 枚舉的方法

枚舉中的方法為每一個枚舉值而“生”。所以倘若想要在特定情況執(zhí)行特定代碼的話,你需要分支處理或采用switch語句來明確正確的代碼路徑。

enum Wearable {
    //枚舉中可以嵌套枚舉
    enum Weight: Int {
        case Light = 1
    }
    enum Armor: Int {
        case Light = 2
    }
    //枚舉值,指定了weight和armor的類型
    case Helmet(weight: Weight, armor: Armor)
    //枚舉方法
    func attributes() -> (weight: Int, armor: Int) {
        switch self {
            case .Helmet(let w, let a):
                return (w.rawValue * 2, a.rawValue * 4)
        }
    }
}
//因為weight和armor都已經(jīng)指定了枚舉類型,直接使用點枚舉值
let wearable = Wearable.Helmet(weight: .Light, armor: .Light)
let woodenHelmetProps = wearable.attributes()
print(woodenHelmetProps) //結(jié)果:(2, 8)

也可以在枚舉中添加靜態(tài)方法,換言之通過一個非枚舉類型來創(chuàng)建一個枚舉。
在這個示例中,我們需要考慮用戶有時將蘋果設(shè)備叫錯的情況(比如AppleWatch叫成iWatch),需要返回一個合適的名稱。

enum Device {
    case AppleWatch
    //添加靜態(tài)方法
    static func fromSlang(term: String) -> Device? {
        if term == "iWatch" {
            return .AppleWatch
        }
        return nil
    }
}
var device = Device.fromSlang("iWatch") //device為 Device? 類型

二、Swift的結(jié)構(gòu)體

在面向過程的編程語言(如C語言)中,結(jié)構(gòu)體用得比較多,但是面向?qū)ο笾螅缭?code>C++和OC中,結(jié)構(gòu)體已經(jīng)很少使用了。這是因為結(jié)構(gòu)體能夠做的事情,類完全可以取而代之。
Swift語言卻非常重視結(jié)構(gòu)體,把結(jié)構(gòu)體作為實現(xiàn)面向?qū)ο蟮闹匾侄巍?code>Swift中的結(jié)構(gòu)體與C++OC中的結(jié)構(gòu)體有很大的差別,C++OC中的結(jié)構(gòu)體只能定義一組相關(guān)的成員變量,而Swift中的結(jié)構(gòu)體不僅可以定義屬性,還可以定義方法。因此,我們可以把Swfit結(jié)構(gòu)體看做是一種輕量級的類。

Swift中類和結(jié)構(gòu)體的共同處在于:
  • 定義屬性用于存儲值
  • 定義方法用于提供功能
  • 定義下標(biāo)腳本用于訪問值
  • 定義構(gòu)造器用于生成初始化值
  • 通過擴展以增加默認實現(xiàn)的功能
  • 實現(xiàn)協(xié)議以提供某種標(biāo)準(zhǔn)功能
Swift中類和結(jié)構(gòu)體的不同處在于:
  • 結(jié)構(gòu)體不具有繼承性
  • 結(jié)構(gòu)體不具備運行時強制類型轉(zhuǎn)換
  • 結(jié)構(gòu)體不具備使用析構(gòu)器的能力
  • 結(jié)構(gòu)體不具備使用引用計的能力
1. 結(jié)構(gòu)體定義
//結(jié)構(gòu)體定義使用struct關(guān)鍵字
struct MarkStruct {
    //結(jié)構(gòu)體也有存儲屬性和計算屬性,這里只定義了存儲屬性
    var mark1: Int 
    var mark2: Int
    var mark3: Int
}
//所有結(jié)構(gòu)體都有一個自動生成的成員逐一初始化構(gòu)造器,用于初始化結(jié)構(gòu)體實例中成員的屬性。
//順序必須和結(jié)構(gòu)體成員順序一致,必須包含所有的成員
var marks = MarkStruct(mark1: 98, mark2: 96, mark3:100)
print(marks.mark1)
print(marks.mark2)
print(marks.mark3)
2. 結(jié)構(gòu)體定義屬性
struct Point{
    var x = 0.0
    var y = 0.0
}
struct MyPoint {
    //定義存儲屬性
    var p = Point() 
    //定義計算屬性
    var point:Point{
        get{
            return p
        }
        set(newPoint){//修改newValue名為newPoint,本質(zhì)還是newValue
            p.x = newPoint.x
            p.y = newPoint.y
        }
    }
}
var p = Point(x:10.0, y:11.0)
var myPoint = MyPoint()
myPoint.point = p
print("x=\(myPoint.point.x),y=\(myPoint.point.y)")
//運行結(jié)果:x=10.0,y=11.0

3. 結(jié)構(gòu)體的方法
//結(jié)構(gòu)體內(nèi)部只有在構(gòu)造函數(shù)(init)中可以修改屬性的值,其他方法內(nèi)不能直接修改結(jié)構(gòu)體內(nèi)部屬性的值。
struct Rect {
    var width:Double
    var height:Double = 0.0
    // 給結(jié)構(gòu)體定義一個方法, 該方法屬于該結(jié)構(gòu)體
    // 結(jié)構(gòu)體中的成員方法必須使用某個實例調(diào)用
    // 成員方法可以訪問成員屬性
    func getWidth() -> Double{
        return width
    }
}
var rect = Rect(width: 10.0, height: 20.0)
// 結(jié)構(gòu)體中的成員方法是和某個實例對象綁定在一起的, 所以誰調(diào)用, 方法中訪問的屬性就屬于誰
print(rect.getWidth())
 
var rect2 = Rect(width: 30.0, height: 20.0)
// 取得rect2這個對象的寬度
print(rect2.getWidth())
4. 值類型和引用類型

值類型被賦予給一個變量、常數(shù)或者本身被傳遞給一個函數(shù)的時候,實際上操作的是值的拷貝。
實際上,在Swift中,所有的基本類型:整數(shù)、浮點數(shù)、布爾值、字符串、數(shù)組和字典,都是值類型,并且都是以結(jié)構(gòu)體的形式在后臺所實現(xiàn)。
Swift中,所有的結(jié)構(gòu)體和枚舉都是值類型。這意味著它們的實例,以及實例中所包含的任何值類型屬性,在代碼中傳遞的時候都會被復(fù)制。

值類型賦值
struct Resolution {
    var width = 0
    var height = 0
}
//創(chuàng)建一個結(jié)構(gòu)體
let hd = Resolution(width: 1920, height: 1080)
//結(jié)構(gòu)體賦值,實際上做的是拷貝操作,cinema和hd結(jié)構(gòu)體在內(nèi)存中各自占用獨立的空間
var cinema = hd
//修改cinema結(jié)構(gòu)體,不會影響hd結(jié)構(gòu)體
cinema.width = 2048
print("cinema is now  \(cinema.width) pixels wide")
//結(jié)果:cinema is now 2048 pixels wide
print("hd is still \(hd.width ) pixels wide")
//結(jié)果:hd is still 1920 pixels wide

與值類型不同,引用類型在被賦予到一個變量、常量或者被傳遞到一個函數(shù)時,操作的并不是其拷貝。因此,引用的是已存在的實例本身而不是其拷貝。
類就是引用類型

引用類型賦值
class ResolutionClass {
    var width = 0
    var height = 0
    init(width: Int, height: Int) {
        self.width = width
        self.height = height
    }
}
//創(chuàng)建一個類對象
let hdClass = ResolutionClass(width: 1920, height: 1080)
//類對象賦值,引用同一個內(nèi)存空間
var cinemaClass = hdClass 
//修改cinema對象,本質(zhì)上也是修改hdClass對象
cinemaClass.width = 2048
print("cinemaClass is now  \(cinemaClass .width) pixels wide")
//結(jié)果:cinema is now 2048 pixels wide
print("hdClass is also \(hdClass.width ) pixels wide")
//結(jié)果:hd is also 2048 pixels wide
5. 類和結(jié)構(gòu)體的選擇

結(jié)構(gòu)體實例總是通過值傳遞,類實例總是通過引用傳遞。這意味兩者適用不同的任務(wù)。當(dāng)你的在考慮一個工程項目的數(shù)據(jù)構(gòu)造和功能的時候,你需要決定每個數(shù)據(jù)構(gòu)造是定義成類還是結(jié)構(gòu)體。

當(dāng)符合一條或多條以下條件時,請考慮構(gòu)建結(jié)構(gòu)體:
  • 結(jié)構(gòu)體的主要目的是用來封裝少量相關(guān)簡單數(shù)據(jù)值。
  • 有理由預(yù)計一個結(jié)構(gòu)體實例在賦值或傳遞時,封裝的數(shù)據(jù)將會被拷貝而不是被引用。
  • 任何在結(jié)構(gòu)體中儲存的值類型屬性,也將會被拷貝,而不是被引用。
  • 結(jié)構(gòu)體不需要去繼承另一個已存在類型的屬性或者行為。
合適的結(jié)構(gòu)體候選者包括:
  • 幾何形狀的大小
  • 一定范圍內(nèi)的路徑
  • 三維坐標(biāo)系內(nèi)一點
有什么問題請在下方評論區(qū)中提出!O(∩_∩)O哈!
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 230,002評論 6 542
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 99,400評論 3 429
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 178,136評論 0 383
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經(jīng)常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,714評論 1 317
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 72,452評論 6 412
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 55,818評論 1 328
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,812評論 3 446
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 42,997評論 0 290
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 49,552評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 41,292評論 3 358
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,510評論 1 374
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 39,035評論 5 363
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 44,721評論 3 348
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 35,121評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,429評論 1 294
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 52,235評論 3 398
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 48,480評論 2 379

推薦閱讀更多精彩內(nèi)容