【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)


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

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

推荐阅读更多精彩内容