除存储属性外,类、结构体和枚举可以定义计算属性,计算属性不直接存储值,而是提供一个 getter 来获取值,一个可选的 setter 来间接设置其他属性或变量的值。
struct Point {
var x = 0.0, y = 0.0
}
struct Size {
var width = 0.0, height = 0.0
}
struct Rect {
var origin = Point()
var size = Size()
var center: Point {
get {
let centerX = origin.x + (size.width / 2)
let centerY = origin.y + (size.height / 2)
return Point(x: centerX, y: centerY)
}
set(newCenter) {
origin.x = newCenter.x - (size.width / 2)
origin.y = newCenter.y - (size.height / 2)
}
}
}
var square = Rect(origin: Point(x: 0.0, y: 0.0),
size: Size(width: 10.0, height: 10.0))
let initialSquareCenter = square.center
square.center = Point(x: 15.0, y: 15.0)
println("square.origin is now at (\(square.origin.x), \(square.origin.y))")
// 输出 "square.origin is now at (10.0, 10.0)”
这个例子定义了 3 个几何形状的结构体:
-
Point
封装了一个(x, y)
的坐标 -
Size
封装了一个width
和height
-
Rect
表示一个有原点和尺寸的矩形
Rect
也提供了一个名为center
的计算属性。一个矩形的中心点可以从原点和尺寸来算出,所以不需要将它以显式声明的Point
来保存。Rect
的计算属性center
提供了自定义的 getter 和 setter 来获取和设置矩形的中心点,就像它有一个存储属性一样。
例子中接下来创建了一个名为square
的Rect
实例,初始值原点是(0, 0)
,宽度高度都是10
。如图所示蓝色正方形。
square
的center
属性可以通过点运算符(square.center
)来访问,这会调用 getter 来获取属性的值。跟直接返回已经存在的值不同,getter 实际上通过计算然后返回一个新的Point
来表示square
的中心点。如代码所示,它正确返回了中心点(5, 5)
。
center
属性之后被设置了一个新的值(15, 15)
,表示向右上方移动正方形到如图所示橙色正方形的位置。设置属性center
的值会调用 setter 来修改属性origin
的x
和y
的值,从而实现移动正方形到新的位置。
便捷 setter 声明
如果计算属性的 setter 没有定义表示新值的参数名,则可以使用默认名称newValue
。下面是使用了便捷 setter 声明的Rect
结构体代码:
struct AlternativeRect {
var origin = Point()
var size = Size()
var center: Point {
get {
let centerX = origin.x + (size.width / 2)
let centerY = origin.y + (size.height / 2)
return Point(x: centerX, y: centerY)
}
set {
origin.x = newValue.x - (size.width / 2)
origin.y = newValue.y - (size.height / 2)
}
}
}
只读计算属性
只有 getter 没有 setter 的计算属性就是只读计算属性。只读计算属性总是返回一个值,可以通过点运算符访问,但不能设置新的值。
<<<<<<< HEAD
注意:
必须使用
var
关键字定义计算属性,包括只读计算属性,因为他们的值不是固定的。let
关键字只用来声明常量属性,表示初始化后再也无法修改的值。注意:
必须使用
var
关键字定义计算属性,包括只读计算属性,因为它们的值不是固定的。let
关键字只用来声明常量属性,表示初始化后再也无法修改的值。a516af6a531a104ec88da0d236ecf389a5ec72af
只读计算属性的声明可以去掉get
关键字和花括号:
struct Cuboid {
var width = 0.0, height = 0.0, depth = 0.0
var volume: Double {
return width * height * depth
}
}
let fourByFiveByTwo = Cuboid(width: 4.0, height: 5.0, depth: 2.0)
println("the volume of fourByFiveByTwo is \(fourByFiveByTwo.volume)")
// 输出 "the volume of fourByFiveByTwo is 40.0"
这个例子定义了一个名为Cuboid
的结构体,表示三维空间的立方体,包含width
、height
和depth
属性,还有一个名为volume
的只读计算属性用来返回立方体的体积。设置volume
的值毫无意义,因为通过width
、height
和depth
就能算出volume
。然而,Cuboid
提供一个只读计算属性来让外部用户直接获取体积是很有用的。