前言
和其他语言不同的是,Swift不需要为自定义的类和结构体创建接口和实现文件。只需要创建单一文件用来创建类和结构体,其他的外部接口的代码系统会自动生成。下面这篇文章主要介绍了关于Swift 3.0类与结构体的内容,感兴趣的朋友一起来看看吧。
类和结构体区别
Swift的类和结构体具有以下相同的特点:
- 可以定义属性来保存值
- 可以定义方法来提供功能
- 可以定义下标来使用他们的值
- 可以定义初始化器来配置他们的初始化状态
- 可以在默认的实现上扩展他们的功能
- 遵从协议来提供标准的功能
类具有结构体没有的额外的功能:
- 继承允许某一个类继承另外一个类的特性
- 类型转换允许你检查并在运行时解释一个类实例的类型
- 析构器允许释放所有该类已经赋值的实例资源
- 引用计数允许多个引用一个类的实例
结构体一般来说赋值的时候是直接拷贝的,没有使用引用计数的机制。
符号定义
下面是一个定义结构体和类的例子:
1
2
3
4
5
6
7
8
9
10
|
struct Resolution {
var width = 0
var height = 0
}
class VideoMode {
var resolution = Resolution()
var interlaced = false
var frameRate = 0.0
var name: String?
}
|
结构体初始化的时候可以直接
1
|
let vga = Resolution(width: 640, height: 480)
|
这点和类不一样,类没有默认的逐个成员的初始化器。
结构体和枚举是值类型
1
2
|
let hd = Resolution(width: 1920, height: 1080)
var cinema = hd
|
再赋值
1
|
cinema.width = 2048
|
结果
1
2
|
print( "cinema is now \(cinema.width) pixels wide" )
// Prints "cinema is now 2048 pixels wide"
|
然而hd.width还是1920
1
2
|
print( "hd is still \(hd.width) pixels wide" )
// Prints "hd is still 1920 pixels wide"
|
可见赋值过程是做了一次深度拷贝。
枚举也是具有同样的行为, 如以下例子,rememberedDirection的值并没有改变:
1
2
3
4
5
6
7
8
9
10
|
enum CompassPoint {
case north, south, east, west
}
var currentDirection = CompassPoint.west
let rememberedDirection = currentDirection
currentDirection = .east
if rememberedDirection == .west {
print( "The remembered direction is still .west" )
}
// Prints "The remembered direction is still .west"
|
类是引用类型
例如:
1
2
3
4
5
|
let tenEighty = VideoMode()
tenEighty.resolution = hd
tenEighty.interlaced = true
tenEighty.name = "1080i"
tenEighty.frameRate = 25.0
|
进行赋值引用
1
2
|
let alsoTenEighty = tenEighty
alsoTenEighty.frameRate = 30.0
|
结果
1
2
|
print( "The frameRate property of tenEighty is now \(tenEighty.frameRate)" )
// Prints "The frameRate property of tenEighty is now 30.0"
|
标识符
- 完全相同(===)
- 不完全相同(!===)
1
2
3
4
|
if tenEighty === alsoTenEighty {
print( "tenEighty and alsoTenEighty refer to the same VideoMode instance." )
}
// Prints "tenEighty and alsoTenEighty refer to the same VideoMode instance."
|
完全相同(===)和等于(==)是不一样的:
- 完全相同意思是两个类类型的常量或者变量指向完全相同的类实例
- 等于意思是两个实例被认为值相同或者相等, 可以自行定义==操作符来进行判断两个实例在某种意义上是相等的
选择使用类和结构体
由于结构体的实例一般是值传递,而类实例一般是引用传递,因此你需要根据实际情况来考虑应该定义一个类还是结构体.
如有以下一种或多仲情况使用结构体:
- 结构体主要的目的是封装少量的相关性简单数据值
- 在结构体的实例赋值或者传递的时候,需要考虑到封装好的值会被拷贝而不是引用是否是合理的
- 任何保存于结构体的属性都是值类型的,他们也是期望被赋值或者传递时是拷贝而不是引用
- 结构体不需要从其他存在的类型继承属性或者行为
看看几个使用结构体恰当的例子:
- 几何图形的大小,可以封装width和height属性,都是Double类型
- 指向连续序列范围的方法,可以封装start和length属性,都是Int类型
- 一个在3D坐标系统的点, 可以封装x, y和z属性,都是Double类型
其他的情况请定义类并创建类实例,管理和传递都使用引用。
在实践中,大部分的自定义数据结构都是使用类居多,很少使用结构体。
String、Array和Dictionary的赋值和拷贝行为
String, Array和 Dictionary都是结构体,因此赋值直接是拷贝,而NSString, NSArray 和NSDictionary则是类,所以是使用引用的方式。
参考英语原文:
https://developer.apple.com/library/content/documentation/Swift/Conceptual/Swift_Programming_Language/ClassesAndStructures.html#//apple_ref/doc/uid/TP40014097-CH13-ID82
总结
以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作能带来一定的帮助,如果有疑问大家可以留言交流,谢谢大家对服务器之家的支持。
原文链接:http://devlong.com/2016/11/02/swift-classes-and-structures/