【Swift 3.1】09 - 类和结构 (Classes and Structures)

类和结构 (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 and Structures))