9、【Swift】类和结构体

  • Swift创建类和结构体时,不需要将声明文件和实现文件分开

一个的实例通常被称为对象。

结构体和类对比

  • 共同点

    • 都有 存储属性
    • 都有函数 / 方法
    • 都有 下标语法
    • 都有 构造器 / 初始化器
    • 都有 拓展
    • 都有 协议
  • 不同点-类多出的功能:

    • 类允许继承,结构体不可继承
    • 类创建的实例,可以在运行时检查、推断类型,结构体只在编译时检查
    • 类有析构器 / 反初始化器,结构体没有
    • 类有引用计数,结构体没有
  • 类比结构体的功能多,也更复杂,建议优先使用结构体

  • 意味着你的大多数自定义数据类型都会是结构体和枚举。更多详细的比较参见 在结构和类之间进行选择

类型定义的语法

  • 类与结构体有相似定义语法
    • 使用关键词 class来定义类
    • 使用struct来定义结构体
struct SomeStructure {
    // 在这里定义结构体
}
class SomeClass {
    // 在这里定义类
}

定义了一个新的类或者结构体:用UpperCamelCase 命名法命名 (比如这里我们说到的 SomeClass和 SomeStructure)以符合 Swift 的字母大写风格(比如说 String , Int 以及 Bool)

属性和方法使用lowerCamelCase命名法[1] (比如 frameRate 和 incrementCount),以此来区别于类型名称

CamelCase names :在给储存器或者函数命名时我们习惯上把多个有意义的单词以开头大写的形式拼接在一起组成一个单一的长单词。这种方法被称为“驼峰式命名法”,又分为开头大写和开头小写两种。比如说 SomeClass 、 frameRate 等。

  • 例子
struct Resolution {
    var width = 0
    var height = 0
  // 两个属性被初始化为整数 0 的时候,它们会被推断为 Int 类型
}
class VideoMode {
    var resolution = Resolution()
    var interlaced = false
    var frameRate = 0.0
    var name: String?
  //  name 是一个可选类型,它会被自动赋予一个默认值 nil,意为“没有 name 值”
}

结构体和类的实例

  • 创建结构体和类实例,语法相似
let someResolution = Resolution()
let someVideoMode = VideoMode()
  • 构造器语法
    • 类型名称后跟随一对空括号
    • 属性均会被初始化为默认值

属性访问

  • 点语法:实例名后面紧跟属性名,两者以点号(.)分隔,不带空格
print("The width of someResolution is \(someResolution.width)")
// 打印 "The width of someResolution is 0"
  • 访问子属性,如 VideoModeresolution 属性的 width 属性
print("The width of someVideoMode is \(someVideoMode.resolution.width)")
// 打印 "The width of someVideoMode is 0"
  • 使用点语法为可变属性赋值:
someVideoMode.resolution.width = 1280
print("The width of someVideoMode is now \(someVideoMode.resolution.width)")
// 打印 "The width of someVideoMode is now 1280"

结构体类型的成员 初始化器 / 构造器

  • Swift的结构体,会自动生成一个成员初始化器
  • 新创建实例的初始化值,通过属性名称传递到成员初始化器中
let vga = Resolution(width: 640, height: 480)

结构体和枚举是值类型

  • 值类型:赋值给变量、常量、传递给函数时,值只会被拷贝
  • 所有的基本类型(底层也是使用结构体实现的):
    • 整数(integer)
    • 浮点数(floating-point number)
    • 布尔值(boolean)
    • 字符串(string)
    • 数组(array)
    • 字典(dictionary)
  • 枚举也是值类型
  • 实例中值类型的属性,代码传递时,都会被复制
  • Copy On Write技术
    • Swift对系统自带的集合类型,进行了优化,新集合不会立即复制,跟原集合共享同一份内存。

    • 只有集合的副本要被修改时,才会复制原集合的元素

  • 例子
let hd = Resolution(width: 1920, height: 1080)
var cinema = hd
  • 因为 Resolution 是结构体,所以会创建一个现有实例的副本,然后将副本赋值给 cinema
  • 修改为稍微宽一点的 2K 标准:
cinema.width = 2048
print("cinema is now  \(cinema.width) pixels wide")
// 打印 "cinema is now 2048 pixels wide"
  • 初始的 hd 实例中 width 属性还是 1920
print("hd is still \(hd.width) pixels wide")
// 打印 "hd is still 1920 pixels wide"
img
  • 枚举也遵循相同的行为准则

类是引用类型

  • 引用类型:赋值到一个常量,变量或者本身被传递到一个函数时,只引用,不拷贝

  • 使用了之前定义的 VideoMode 类:

let tenEighty = VideoMode()
tenEighty.resolution = hd
tenEighty.interlaced = true
tenEighty.name = "1080i"
tenEighty.frameRate = 25.0
  • tenEighty 赋值给一个名为 alsoTenEighty 的新常量,并修改 alsoTenEighty 的帧率:
let alsoTenEighty = tenEighty
alsoTenEighty.frameRate = 30.0
  • 类是引用类型,所以 tenEightalsoTenEight 实际上引用的是同一个 VideoMode 实例。换句话说,它们是同一个实例的两种叫法
img
  • 两个实例虽然都是常量,但不存储 VideoMode 实例,只存储它的引用 / 指针,改变的是实例的属性,属性是变量

恒等运算符

  • 恒等预算符:判断两 类创建的实例 是否是引用同一个
    • 相同(===)// Identical to
    • 不相同(!==)// Not identical to
if tenEighty === alsoTenEighty {
    print("tenEighty and alsoTenEighty refer to the same VideoMode instance.")
}
// 打印 "tenEighty and alsoTenEighty refer to the same VideoMode instance."
  • 判断引用类型:指针地址是否相等,用 ===
  • 判断值类型:数值是否相等,用 ==
  • 在章节 等价操作符 中将会详细介绍实现自定义 == 和 != 运算符的流程

指针

  • 和 C,C++ 或者 Objective-C 语言不同,Swift指针不使用星号(*),跟普通常量、变量一样即可
  • 需要操作指针,用标准库提供的指针和缓冲区类型 —— 参见章节 手动管理内存

类和结构体之间的选择

  • 结构体-使用场景:
    • 只封装简单数据
    • 传递数据时,只需要数据值,而不是引用
    • 数据的属性也是值类型,值需要拷贝,不需要引用
    • 不需要继承
  • 结构体-使用例子:
    • 几何形状大小,width属性和 height属性,两者都为 double类
    • 路径,封装了一个 start属性和 length属性,两者为 Int类型
    • 三维坐标系,封装了 x , y 和 z属性,他们都是 double类型

字符串,数组和字典的赋值与拷贝行为

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