结构体(三)

2018-05-14  本文已影响0人  小橘子成长记

结构体值

当涉及到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标准库中定义的协议。

上一篇下一篇

猜你喜欢

热点阅读