06 Swift 闭包\枚举\结构体

1. 闭包

** 闭包**
函数是闭包的一种
类似于OC语言中block
闭包的表达式(匿名函数)--能够捕获上下文中的值

语法:in 关键字的目的是便于区分返回值和执行语句
闭包的表达式类型和函数的类型一样,是参数加上返回值,也就是in 之前的一部分
{
(参数)-> 返回值类型 in
执行语句

}

完整的写法:

let say:(String) -> Void = {
    (name: String) -> Void in
    print("hi \(name)")

}
say("gaojun")

let say:(String) -> Void = {
    (name: String) -> Void in
    print("hi \(name)")

}
say("gaojun")

闭包表达式作为回调函数

传统数组排序写法

缺点:不一定是由小到大,不一定是全部比较,有可能只比较个位数,所以如何比较可以交给调用者决定

func bubbleSort (inout array:[Int])
{
    let count = array.count

    for var i = 1 ; i < count ; i++
    {
        for var j = 0 ; j < (count - i) ; j++
        {
            if array[j] > array[j + 1]
            {
                let temp = array[j]
            
                array[j] = array[j + 1]
            
                array[j + 1] = temp
            }
        }
    }
}

闭包写法

  func bubbleSort(inout array:[Int], cmp: (Int, Int) ->Int)
{
    let count = array.count

    for var i = 1 ; i < count ; i++
    {
        for var j = 0 ; j < (count - i) ; j++
        {
            if cmp(array[j], array[j + 1]) == -1
            {
        
                let temp = array[j]
            
                array[j] = array[j + 1]
            
                array[j + 1] = temp
            }
        }
    }
}

let cmp = {

    (a: Int, b: Int) -> Int in
    if a > b{
        return 1;
    }else if a < b {
    
        return -1

    }else{

        return 0
  }

}

var arr: Array<Int> = [31, 13, 52, 84, 5]

bubbleSort(&arr, cmp: cmp)

print(arr)

闭包作为参数传递

var arr1:Array<Int> =  [31, 13, 52, 84, 5]

bubbleSort(&arr1, cmp: {

    (a: Int, b: Int) -> Int in
    if a > b{
        return 1;
    }else if a < b {
    
        return -1
    
    }else{
    
        return 0
    }

})

** 尾随闭包**

如果闭包是最后一个参数,可以直接将闭包写到参数列表后面
这样可以提高阅读性,称之为尾随闭包

bubbleSort(&arr1
    ){

    (a: Int, b: Int) -> Int in
    if a > b{
        return 1;
    }else if a < b {
    
        return -1
    
    }else{
    
        return 0
    }
}

闭包表达式优化

1、类型优化,由于函数中已经声明了闭包的参数类型,所以传入的实参可以不用写类型

2、返回值优化,同理由于函数中已经声明了闭包的返回值类型,所以传入的实参可以不用写类型
3、参数优化,Swift可以使用$索引的方式来访问闭包的参数

bubbleSort(&arr1){

    if $0 > $1{

        return 1
    }else if $0 < $1{

        return -1
    }else{

        return 0
    }
}

 // 如果只有一条语句可以省略return

let hehe = {

    "我是gaojun"
}

** 闭包捕获值**

func getIncFunc() -> (Int) ->Int
{
    var max = 10

    func incFunc (x: Int) ->Int{

        print("incFunc函数结束")
    
        max++
    
        return max +  x
    }

/*
    当执行到这一句时inc参数就应该被释放了
    但是由于在内部函数中使用到了它,所以它被捕获了
    同理,当执行完这一句时max变量被释放了
    但是由于在内部函数中使用到了它,所以它被捕获了
*/

print("getIncFunc函数结束")
return incFunc
}

//被捕获的值回和与之对应方法绑定在一起 同一个方法中的变量会被绑定到不同的方法中

let incFunc = getIncFunc()

print("--------")

print(incFunc(5))

print("--------")

print(incFunc(5))

**输出结果: **

Snip20160301_6.png

2. 枚举

Swift中的枚举比OC中的枚举强大,因为Swift中的枚举是一等类型
它可以像类和结构体一样增加属性和方法
格式:

enum Method{

    case 枚举值
}

enum Method {

case Add, Sub, Mul, Div

}

// 可以使用枚举类型变量或常量接受枚举值

var m: Method = .Add

// 注意:如果变量或常量没有指定类型,那么前面必须加上该值属于哪个枚举类型

var m1 = Method.Add

// 利用Switch匹配
// 如果case中包含所有的值,可以不写default;如果case没有包含所有的值,必须写

switch (Method.Add){

case Method.Add:
    print("加法")
case Method.Sub:
    print("减法")
case Method.Mul:
    print("除法")
case Method.Div:
    print("乘法")
}

// 原始值
// OC中枚举的本质就是整数,所以OC中的枚举是有原始值的,默认是从0开始
// 而Swift中的枚举默认是没有原始值得,但是可以在定义时告诉系统让枚举有原始值

enum Method : 枚举值原始值类型{
    case 枚举值

}

enum Method2: Int{
    可以写在一起
    case Add, Sub, Mul, Div
}

和OC中的枚举一样,也可以指定原始值,后面的值默认 +1
enum Method3 : Int{
    case Add = 5, Sub, Mul, Div

}

Swift 中的枚举除了可以指定整型意外还可以指定其他类型
但是如果指定其他类型,必须给所有枚举值赋值,因为不能自动递增

  enum Method: Double{

    case Add = 5.0, Sub = 6.0, Mul = 6.1, Div = 8.0
}

rawValue代表将枚举值转换成原始值,注意老版本中转换原始值的方法名叫toRaw

Method4.Sub.rawValue

原始值转换为枚举值

enum Method5: String{
    case Add = "add", Sub = "sub", Mul = "mul", Div = "div"

}

通过原始值创建枚举值
注意:
1、原始值区分大小写
2、返回值的是一个可选值,因为原始值对应的枚举值不一定存在
3、老版本中为fromRaw(“add”)

enum Method5: String{
    case Add = "add", Sub = "sub", Mul = "mul", Div = "div"

}

let m2 = Method5(rawValue:"add")

print(m2)

func chooseMethod(op:String)
{
    // 由于返回值是可选类型,所以有可能为nil,最好使用可选绑定
    if let opE = Method5(rawValue: op){
        switch(opE){
        case .Add:
            print("加法")
        case .Sub:
            print("减法")
        case .Mul:
            print("除法")
        case .Div:
              print("乘法")
        
        }
    }
}

枚举相关值:

// 可以让枚举值对应的原始值不是唯一的,而是一个变量
// 每一个枚举可以是在某种模式下的一些特定值

enum lineSegmentDescriptor{

case StartAndEndPattern(start: Double, end: Double)
case StartAndLengthPattern(start: Double, length: Double)

}

var lsd =     lineSegmentDescriptor.StartAndEndPattern(start: 0.0, end: 50.0)

lsd = lineSegmentDescriptor.StartAndLengthPattern(start: 0.0, length: 100.0)

利用switch提取枚举联值

switch lsd
{
    case let .StartAndEndPattern(s, e):
        print("start = \(s) end = \(e)")
    case .StartAndLengthPattern(let s, let l):
        print("start = \(s) length = \(l)")
}

**输出结果: **

Snip20160301_7.png

3. 结构体

结构体

结构体是用于封装不同或相同类型的数据的
Swift中的结构体是一类类型,可以定义属性和方法(甚至构造方法和析构方法等)
格式:

struct 结构体名称 {
    结构体属性和方法

}

struct Rect {

    var width: Double = 0.0

    var height: Double = 0.0

}

如果结构体的属性有默认值,可以直接使用()构造一个结构体
如果结构体的属性没有默认值,必须使用逐一构造器实例化结构体

var r = Rect()

print("width = \(r.width) height = \(r.height)")

*结构体属性的访问使用.语法 *

struct Rect {

var width: Double = 0.0

var height: Double = 0.0

}

var r = Rect()

r.width = 100
r.height = 99

print("width = \(r.width) height = \(r.height)")

结构体构造器
Swift中的结构体和类跟其他面向对象语言一样都有构造函数,而OC是没有的
Swift要求实例化一个结构体或类的时候,所有的成员变量必须有初始值
构造函数的意义就是用于初始化所有成员变量的,而不是分配内存的,分配内存是系统帮我们做的
如果结构体中所有属性都有默认值,可以调用()构造一个结构体实例
如果结构体中的属性没有默认值,可以自定义构造器,并在构造器中给所欲的属性赋值
其实结构体有一个默认的逐一构造器,用于初始化时果所有属性赋值

struct Rect2 {
    var width: Double
    var height: Double = 0.0
}

逐一构造器

var r1 = Rect2(width: 10.0, height: 10.0)

// 错误写法,书写的顺序必须与结构体中成员的顺序一致
// 错误写法,书写必须包含所有成员

结构体中定义成员方法

// 在C和OC中结构体只有属性,而Swift中结构体中还可以定义方法

struct Rect3 {
    var width: Double
    var height: Double = 0.0

// 给结构体定义一个方法,该方法属于该结构体
// 结构体中的成员方法必须使用某个实例调用
// 成员方法可以反复顾问成员属性
func getWidth() -> Double{

        return width
    }
}

var r2 = Rect3(width: 10.0, height: 20.0)

 // 结构体中的成员方法是和某个实例对象绑定在一起的,所以谁调用,方法中访问的属性就属于谁
print(r2.getWidth())

var r3 = Rect3(width: 30.0, height: 20.0)

print(r3.getWidth())

结构体是值类型

struct Rect4 {
    var width: Double
    var height: Double = 0.0

    func show() ->Void{

        print("width = \(width) height = \(height)")

    }
}

var r4 = Rect4(width: 10.0, height: 10.0)

var r5 = r4

r4.show()

r5.show()

r4.width = 20.0

 // 结构体是值类型,结构体之间的赋值其实是将r4中的值完全拷贝一份到r5中,所以他们是两个不同的实例
r4.show()
r5.show()

作者说:

由于这学期的课程实在是太满了, 所以在写笔记的时候, 是直接复制我在学习的时候的笔记, 没有加以整理. 望各位见谅.
由于我之前没有学习Swift中关于协议内存管理以及使用Foundation框架现在我在抽空学习, 过一段时间会更新的.

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,547评论 6 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,399评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,428评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,599评论 1 274
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,612评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,577评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,941评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,603评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,852评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,605评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,693评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,375评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,955评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,936评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,172评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,970评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,414评论 2 342

推荐阅读更多精彩内容