结构体值
当涉及到Swift的结构时,值具有重要的意义,这是因为结构创建了所谓的值类型。
值类型是在赋值时复制实例的类型。
var a = 5
var b = a
print(a) // 5
print(b) // 5
a = 10
print(a) // 10
print(b) // 5
这个"分配时复制"行为意味着当a分配给b时,a的值被复制到b中。
var area1 = DeliveryArea(center: Location(x: 2, y: 4), radius: 2.5)
var area2 = area1
print(area1.radius) // 2.5
print(area2.radius) // 2.5
area1.radius = 4
print(area1.radius) // 4.0
print(area2.radius) // 2.5
和前面的例子一样,area2.radius 没有取到area1半径的新值。这说明使用了结构的值语义。当你分配area1的值时,它得到的这个值是一个拷贝的值。area1和area2仍然完全独立!
结构随处可见
你看到了Location结构和一个简单的Int都有相同的“分配时复制”行为。这是因为它们都是值类型,都具有值语义。
你知道结构代表值,那么Int是什么呢?如果你
看看Swift library中Int的定义,你可能会有点惊讶:
public struct Int : FixedWidthInteger, SignedInteger {
// ...
}
Int类型也是一个结构体。事实上,许多标准的Swift类型都是结构体:Double、String、Bool、Array和Dictionary都为结构体。struct的值语义提供了许多的优点,所以它代表Swift的核心类型。
遵循协议
你可能已经发现上面的Swift library中对Int定义的一些不熟悉的部分。类型FixedWidthInteger和SignedInteger在Int的声明之后出现:
public struct Int : FixedWidthInteger, SignedInteger {
// ...
}
这些类型称为协议。在声明了Int之后,将它们放在冒号之后,你声明的Int类型将符合这些协议。
协议包含一组需求,遵循协议的类型必须满足这些需求。标准库中一个简单的例子CustomStringConvertible:
public protocol CustomStringConvertible {
/// A textual representation of this instance.
public var description: String { get }
}
该协议包含一个属性要求: description(描述)。此描述实际上是“此实例的文本表示”。
如果您要修改DeliveryArea以遵循CustomStringConvertible协议,您将需要添加一个description(描述)属性,并带有实例的“文本表示”。现在试试这个。把DeliveryArea更改为:
struct DeliveryArea: CustomStringConvertible {
let center: Location
var radius: Double
var description: String {
return """
Area with center: x: \(center.x) - y: \(center.y),
radius: \(radius),
"""
}
func contains(_ location: Location) -> Bool {
let distanceFromCenter =
distance(from: (center.x, center.y),
to: (location.x, location.y))
return distanceFromCenter < radius
} }
description(描述)属性的值基于center(中心)和当前radius(半径)。这是通过将其作为计算属性实现的。
你以后将了解到所有关于计算属性的知识。
那么,遵循协议的具体做法是什么呢?因为任何遵循CustomStringConvertible的类型都必须定义description(描述),你可以在任何遵循CustomStringConvertible的类型的实例上调用description(描述)。Swift标准库利用这一点和print()函数。该函数可在控制台上使用自己定义description(描述),而不是一个默认描述:
print(area1) // Area with center: x: 2 - y: 4, radius: 4.0
print(area2) // Area with center: x: 2 - y: 4, radius: 2.5
任何命名类型都可以使用协议来扩展它们的行为。在这种情况下,你的结构要遵循在Swift标准库中定义的协议。