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

2017-01-24  本文已影响110人  Lebron_James

类和结构 (Classes and Structures)

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


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

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

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

类拥有结构没有的功能:

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

定义语法 (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)

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

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

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

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

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

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


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


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

上一篇下一篇

猜你喜欢

热点阅读