Swift5.4 语言指南(十一) 结构和类
★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★
➤微信公众号:山青咏芝(shanqingyongzhi)
➤博客园地址:山青咏芝(https://www.cnblogs.com/strengthen/)
➤GitHub地址:https://github.com/strengthen/LeetCode
➤原文地址:https://www.cnblogs.com/strengthen/p/9729270.html
➤如果链接不是山青咏芝的博客园地址,则可能是爬取作者的文章。
➤原文已修改更新!强烈建议点击原文地址阅读!支持作者!支持原创!
★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★
结构和类是通用的,灵活的结构,它们成为程序代码的构建块。您可以定义属性和方法,以使用与定义常量,变量和函数相同的语法向结构和类添加功能。
与其他编程语言不同,Swift不需要您为自定义结构和类创建单独的接口和实现文件。在Swift中,您可以在一个文件中定义一个结构或类,并且该类或结构的外部接口会自动提供给其他代码使用。
笔记
传统上将类的实例称为对象。然而,夫特结构和类在功能上更接近比在其他语言,等等本章的描述适用于实例功能任一类或结构类型。因此,使用了更通用的术语实例。
比较结构和类
Swift中的结构和类有很多共同点。两者都可以:
- 定义属性以存储值
- 定义提供功能的方法
- 定义下标以使用下标语法提供对它们的值的访问
- 定义初始值设定项以设置其初始状态
- 扩展以扩展其功能,超出默认实现
- 符合协议以提供某种标准功能
类具有结构没有的其他功能:
- 继承使一个类可以继承另一个类的特征。
- 通过类型转换,您可以在运行时检查和解释类实例的类型。
- 反初始化程序使类的实例可以释放其已分配的所有资源。
- 引用计数允许对一个类实例进行多个引用。
有关更多信息,请参见继承,类型转换,取消初始化和自动引用计数。
类支持的其他功能是以增加复杂性为代价的。作为一般准则,应首选结构,因为它们更易于推理,并在适当或必要时使用类。实际上,这意味着您定义的大多数自定义数据类型将是结构和枚举。有关更详细的比较,请参见在结构和类之间进行选择。
定义语法
结构和类具有相似的定义语法。您可以使用struct
关键字介绍结构,并使用关键字介绍类class
。两者都将它们的整个定义放在一对大括号内:
- struct SomeStructure {
- // structure definition goes here
- }
- class SomeClass {
- // class definition goes here
- }
笔记
每当定义新的结构或类时,就定义新的Swift类型。给出类型UpperCamelCase
名称(如SomeStructure
和SomeClass
这里)来匹配标准斯威夫特类型的资本(如String
,Int
和Bool
)。给属性和方法lowerCamelCase
名称(例如frameRate
和incrementCount
)以将它们与类型名称区分开。
这是一个结构定义和类定义的示例:
- struct Resolution {
- var width = 0
- var height = 0
- }
- class VideoMode {
- var resolution = Resolution()
- var interlaced = false
- var frameRate = 0.0
- var name: String?
- }
上面的示例定义了一个称为的新结构Resolution
,用于描述基于像素的显示分辨率。此结构具有两个存储的属性,分别称为width
和height
。存储的属性是常量或变量,它们捆绑在一起并存储为结构或类的一部分。通过将这两个属性Int
设置为的初始整数值,可以推断出这两个属性的类型0
。
上面的示例还定义了一个名为的新类VideoMode
,以描述用于视频显示的特定视频模式。此类具有四个变量存储的属性。首先resolution
使用新的Resolution
结构实例初始化,该实例推断属性类型为Resolution
。对于其他三个属性,新VideoMode
实例将使用interlaced
设置false
(表示“非隔行视频”),回放帧速率0.0
以及String
称为的可选值来初始化name
。该name
属性会自动获得默认值nil
或“无name
值”,因为它是可选类型。
结构和类实例
该Resolution
结构定义和VideoMode
类定义只是描述一个东西Resolution
或VideoMode
看起来像。它们本身并没有描述特定的分辨率或视频模式。为此,您需要创建结构或类的实例。
对于结构和类,创建实例的语法非常相似:
- let someResolution = Resolution()
- let someVideoMode = VideoMode()
结构和类都对新实例使用初始化程序语法。初始化程序语法的最简单形式是使用类或结构的类型名称,后跟空括号,例如Resolution()
或VideoMode()
。这将创建类或结构的新实例,并将所有属性初始化为其默认值。类和结构的初始化在Initialization中有更详细的描述。
访问属性
您可以使用点语法访问实例的属性。在点语法中,您应在实例名称之后立即写上属性名称,并用句点(.
)分隔,不能有任何空格:
- print("The width of someResolution is \(someResolution.width)")
- // Prints "The width of someResolution is 0"
在此示例中,someResolution.width
引用的width
属性someResolution
,并返回其默认初始值0
。
您可以向下钻取子属性,例如的width
属性中的resolution
属性VideoMode
:
- 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"
结构类型的成员初始化器
所有结构都有一个自动生成的逐成员初始化器,您可以使用该初始化器来初始化新结构实例的成员属性。可以通过名称将新实例的属性的初始值传递给成员初始化器:
- let vga = Resolution(width: 640, height: 480)
与结构不同,类实例不接收默认的成员初始化器。初始化在初始化中有更详细的描述。
结构和枚举是值类型
甲值类型是一个类型,其值被复制时,它的传递给一个函数时,它的分配给一个变量或常数,或。
实际上,在前几章中,您一直在广泛使用值类型。实际上,Swift中的所有基本类型(整数,浮点数,布尔值,字符串,数组和字典)都是值类型,并被实现为幕后结构。
所有结构和枚举都是Swift中的值类型。这意味着您创建的任何结构和枚举实例以及它们具有的任何值类型作为属性,都将在它们在代码中传递时始终被复制。
笔记
由标准库(如数组,字典和字符串)定义的集合使用优化来降低复制的性能成本。这些集合不共享立即复制的功能,而是共享存储在原始实例与任何副本之间的元素的内存。如果集合的副本之一被修改,则元素将在修改之前被复制。您在代码中看到的行为始终就像是立即进行了复制一样。
考虑以下示例,该示例使用Resolution
上一个示例的结构:
- let hd = Resolution(width: 1920, height: 1080)
- var cinema = hd
本示例声明一个常量hd
,并将其设置为使用Resolution
全高清视频的宽度和高度(1920像素宽x 1080像素高)初始化的实例。
然后,它声明一个名为的变量cinema
,并将其设置为的当前值hd
。因为Resolution
是结构,所以将创建现有实例的副本,并将此新副本分配给cinema
。尽管现在hd
和cinema
现在具有相同的宽度和高度,但它们是幕后的两个完全不同的实例。
接下来,将的width
属性cinema
修改为用于数字电影放映的稍宽的2K标准的宽度(宽2048像素,高1080像素):
- cinema.width = 2048
检查的width
属性cinema
表明它确实已更改为2048
:
- print("cinema is now \(cinema.width) pixels wide")
- // Prints "cinema is now 2048 pixels wide"
但是,width
原始hd
实例的属性仍然具有旧值1920
:
- print("hd is still \(hd.width) pixels wide")
- // Prints "hd is still 1920 pixels wide"
当cinema
提供了的当前值时hd
,将其中存储的值hd
复制到新cinema
实例中。最终结果是两个完全独立的实例,其中包含相同的数值。然而,因为它们是独立的情况下,设定的宽度cinema
,以2048
不影响存储在所述宽度hd
,如示于下图中:
相同的行为适用于枚举:
- enum CompassPoint {
- case north, south, east, west
- mutating func turnNorth() {
- self = .north
- }
- }
- var currentDirection = CompassPoint.west
- let rememberedDirection = currentDirection
- currentDirection.turnNorth()
- print("The current direction is \(currentDirection)")
- print("The remembered direction is \(rememberedDirection)")
- // Prints "The current direction is north"
- // Prints "The remembered direction is west"
当rememberedDirection
被赋予的价值currentDirection
,它实际上是设置为该值的副本。更改currentDirection
其后的值不会影响存储在中的原始值的副本rememberedDirection
。
类是引用类型
不像值类型,引用类型是不当他们分配给他们传递给函数的可变或恒定的,或者当复制。而不是副本,而是使用对相同现有实例的引用。
这是一个使用VideoMode
上面定义的类的示例:
- let tenEighty = VideoMode()
- tenEighty.resolution = hd
- tenEighty.interlaced = true
- tenEighty.name = "1080i"
- tenEighty.frameRate = 25.0
本示例声明一个名为的新常量tenEighty
,并将其设置为引用VideoMode
该类的新实例。视频模式被分配了1920
by1080
之前的HD分辨率的副本。将其设置为隔行扫描,将其名称设置为"1080i"
,并将其帧速率设置为25.0
每秒的帧数。
接下来,tenEighty
将分配给一个称为的新常数alsoTenEighty
,并alsoTenEighty
修改的帧频:
- let alsoTenEighty = tenEighty
- alsoTenEighty.frameRate = 30.0
因为类是引用类型,tenEighty
并且alsoTenEighty
实际上两者都引用同一个 VideoMode
实例。实际上,它们只是同一单个实例的两个不同名称,如下图所示:
检查的frameRate
属性tenEighty
表明它正确地报告30.0
了基础VideoMode
实例的新帧速率:
- print("The frameRate property of tenEighty is now \(tenEighty.frameRate)")
- // Prints "The frameRate property of tenEighty is now 30.0"
此示例还显示了如何更难以推理引用类型。如果tenEighty
和alsoTenEighty
在程序代码中相距甚远,则可能很难找到更改视频模式的所有方式。无论在哪里使用tenEighty
,都还必须考虑使用的代码,alsoTenEighty
反之亦然。相反,值类型更容易推论,因为与相同值交互的所有代码在源文件中都在一起。
请注意,tenEighty
和alsoTenEighty
被声明为常量,而不是变量。但是,您仍然可以更改tenEighty.frameRate
,alsoTenEighty.frameRate
因为tenEighty
和alsoTenEighty
常量本身的值实际上并未更改。tenEighty
并且alsoTenEighty
它们自己并不“存储”该VideoMode
实例,相反,它们都引用VideoMode
了幕后的实例。更改frameRate
的是基础的属性VideoMode
,而不是对该常量的常量引用的值VideoMode
。
身份运算符
因为类是引用类型,所以多个常量和变量有可能在后台引用类的同一单个实例。(对于结构和枚举,情况并非如此,因为在将它们分配给常量或变量或传递给函数时,它们始终会被复制。)
有时找出两个常量或变量是否引用了类的完全相同的实例有时会很有用。为此,Swift提供了两个身份运算符:
- 与(
===
)相同 - 与(
!==
)不同
使用这些运算符检查两个常量或变量是否引用相同的单个实例:
- if tenEighty === alsoTenEighty {
- print("tenEighty and alsoTenEighty refer to the same VideoMode instance.")
- }
- // Prints "tenEighty and alsoTenEighty refer to the same VideoMode instance."
需要注意的是相同的(由三个代表等号,或===
)并不意味着作为同一件事等于(由两个代表等号,或==
)。等同于表示类类型的两个常量或变量引用完全相同的类实例。等于表示两个实例的值相等或相等,对于类型equals来说,这是类型设计者所定义的。
当您定义自己的自定义结构和类时,您有责任确定两个实例相等时的资格。等价运算符中介绍了定义自己的==
和!=
运算符实现的过程。
指针
如果您有使用C,C ++或Objective-C的经验,您可能会知道这些语言使用指针来引用内存中的地址。引用某种引用类型的实例的Swift常量或变量类似于C中的指针,但不是指向内存中地址的直接指针,并且不需要您写星号(*
)来表示您正在创建参考。相反,这些引用的定义与Swift中的其他任何常量或变量一样。标准库提供了指针和缓冲区类型,如果您需要直接与指针进行交互,可以使用它们(请参见《手动内存管理》)。