美文网首页
【Swift 3.1】09 - 类和结构 (Classes an

【Swift 3.1】09 - 类和结构 (Classes an

作者: Lebron_James | 来源:发表于2017-01-24 20:38 被阅读110次

    类和结构 (Classes and Structures)

    自从苹果2014年发布Swift,到现在已经两年多了,而Swift也来到了3.1版本。去年利用工作之余,共花了两个多月的时间把官方的Swift编程指南看完。现在整理一下笔记,回顾一下以前的知识。有需要的同学可以去看官方文档>>


    在Swift中,Swift的类和结构不需要创建接口和实现两个文件,只需要在一个文件中定义类或者结构,外部接口会自动生成给其他代码使用。

    类和结构的对比 (Comparing Classes and Structures)

    Swift的类和结构有很多共同点:

    • 定义属性来存储值
    • 定义提供某些功能的方法
    • 定义下标来通过下标访问它们的值
    • 定义初始化器来设置他们的初始状态
    • 可以扩展来扩展默认实现没有的功能
    • 可以遵循协议来提供某种标准化的功能

    类拥有结构没有的功能:

    • 从其他类中继承
    • 类型转换
    • 类的实例可以使用Deinitializers清除他自己定义的资源
    • 引用计数允许一个类实例有一个或多个引用

    注意:结构在使用中被传递时,是通过复制的方式,不是引用计数。

    定义语法 (Definition Syntax)

    通用形式如下:

    class SomeClass {
        // class definition goes here
    }
    struct SomeStructure {
        // structure definition goes here
    }
    

    下面是一个例子:

    struct Resolution {
        var width = 0
        var height = 0
    }
    class VideoMode {
        var resolution = Resolution()
        var interlaced = false
        var frameRate = 0.0
        var name: String?
    }
    
    类和结构的实例 (Class and Structure Instances)

    创建实例的语法:

    let someResolution = Resolution()
    let someVideoMode = VideoMode()
    

    通过这种方式创建的实例,实例的属性都初始化为默认值。

    访问属性 (Accessing Properties)

    使用点语法访问属性:

    print("The width of someResolution is \(someResolution.width)")
    // Prints "The width of someResolution is 0"
    
    print("The width of someVideoMode is \(someVideoMode.resolution.width)")
    // Prints "The width of someVideoMode is 0"
    

    使用点语法给变量赋值:

    someVideoMode.resolution.width = 1280
    print("The width of someVideoMode is now \(someVideoMode.resolution.width)")
    // Prints "The width of someVideoMode is now 1280"
    

    注意:不同于OC,Swift允许我们直接修改结构属性的子属性,在上面的例子中,someVideoModeresolution属性的width属性是直接设置的,无需设置整个resolution属性的值。

    结构类型的成员逐一始化器 (Memberwise Initializers for Structure Types)

    所有结构都有一个自动生成的成员逐一始化器,我们可以用来初始化成员属性的值。

    let vga = Resolution(width: 640, height: 480)
    

    不同于结构,类是没有默认的初始化器的。

    结构和枚举都是值类型 (Structures and Enumerations Are Value Types)

    值类型就是当它被赋值给变量或常量,或者被当做参数传入方法时,是通过赋值的方式实现的。

    实际上,Swift的基本类型,例如整型、浮点型、布尔值、字符串、数组和字典都是是值类型,底层都是通过结构的实现的。

    let hd = Resolution(width: 1920, height: 1080)
    var cinema = hd
    

    hdcinema有同样的高度和宽度,但是这两个是完全不一样的实例。

    cinema.width = 2048
    print("cinema is now \(cinema.width) pixels wide")
    // Prints "cinema is now 2048 pixels wide"
    
    print("hd is still \(hd.width) pixels wide")
    // Prints "hd is still 1920 pixels wide"
    

    cinemawidth属性改为2048,但是hdwidth属性还是1920。

    类是引用类型 (Classes Are Reference Types)

    不同于值类型,引用类型被赋值给变量或常量,或者被当做参数传入方法时,不是通过复制实现的,而是引用同一个实例。

    let tenEighty = VideoMode()
    tenEighty.resolution = hd
    tenEighty.interlaced = true
    tenEighty.name = "1080i"
    tenEighty.frameRate = 25.0
    
    let alsoTenEighty = tenEighty
    alsoTenEighty.frameRate = 30.0
    

    因为类是引用类型,实际上tenEightyalsoTenEighty引用着同一个VideoMode实例。tenEightyalsoTenEighty.frameRate也是30.0。

    print("The frameRate property of tenEighty is now \(tenEighty.frameRate)")
    // Prints "The frameRate property of tenEighty is now 30.0"
    

    注意:tenEightyalsoTenEighty是常量,但是我们任然可以修改frameRate属性,因为tenEightyalsoTenEighty的值实际上没有改变。tenEightyalsoTenEighty不是存储VideoMode实例,而是引用着同一个VideoMode实例。在底层里实际上是VideoMode实例的frameRate属性在改变,而不是引用着VideoMode实例的常量改变。

    相同运算符 (Identity Operators)

    使用下面两个运算符来判断两个常量或变量是否引用着同一个实例:

    • 等于 (===)
    • 不等于 (!==)
    if tenEighty === alsoTenEighty {
        print("tenEighty and alsoTenEighty refer to the same VideoMode instance.")
    }
    // Prints "tenEighty and alsoTenEighty refer to the same VideoMode instance."
    

    注意:=====是不一样的:

    • ===意味着两个变量引用着同一个类实例。
    • ==是用来判断两个值是否相等
    指针 (Pointers)

    如果你有C、C++或者是OC的经验,你肯定会知道这些语言是使用指针来引用内存地址。Swift的常量或变量引用一个类实例类似于C语言的指针,但不是一个直接的指针指向内存地址,而且也不需要使用*来暗示我们正在创建一个引用。

    类和结构的选择 (Choosing Between Classes and Structures)

    我们可以使用类和结构来定义数据类型,但是结构是值类型,而类是引用类型。

    在下列的情况下,考虑使用结构:

    • 结构最初的目的是用来封装一些相关并简单的数据
    • 封装的值在被复制给其他变量或者方法时,是通过复制的方式而不是引用
    • 结构的所有属性都是值类型,通过复制的方式更合理
    • 结构不需要集成其他类型的属性或方法

    下面是非常适合用结构的场景:

    • 几何形状的尺寸,也许要封装widthheight属性,都是Double类型
    • 在一个系列类引用一个范围的一个方式,封装startlength属性,都是Int类型
    • 3D坐标系统的一个点,封装xyz属性,都是Double类型

    字符串、数组和字典的赋值和赋值行为

    在Swift中,很多基本数据类型,例如StringArrayDictionary都是用结构实现的。也就意味着他们被赋值给一个常量或者变量、或者被传入一个方法时是通过复制的方式实现的。

    这个不同于NSStringNSArrayNSDictionary,他们是通过类实现的,不是结构。他们被赋值给一个常量或者变量、或者被传入一个方法时是通过引用的方式实现的,不是复制。


    第九部分完。下个部分:【Swift 3.1】10 - 属性 (Properties)


    如果有错误的地方,欢迎指正!谢谢!

    相关文章

      网友评论

          本文标题:【Swift 3.1】09 - 类和结构 (Classes an

          本文链接:https://www.haomeiwen.com/subject/bhxtittx.html