内容简介:作者:Thomas Hanning,译者:Sunnyyoung;校对:Swift 中有两种类型的属性:存储属性与计算属性。存储属性将值(常量或者变量)保存为实例或类型的一部分,而计算属性没有存储值。
作者:Thomas Hanning, 原文链接 ,原文日期:2018-03-15
译者:Sunnyyoung;校对: 小铁匠Linus , numbbbbb ;定稿: Forelax
Swift 中有两种类型的属性:存储属性与计算属性。存储属性将值(常量或者变量)保存为实例或类型的一部分,而计算属性没有存储值。
提示:这篇文章已经更新至 Swift 4。
存储属性
让我们从存储属性开始看起。想象一下你有一个名为 Circle 的类:
class Circle { var radius: Double = 0 } let circle = Circle() circle.radius = 10 print("radius: \(circle.radius)") //radius: 10.0
Circle 拥有名为 radius
的实例变量,默认值为 0。在 Swift 中,每个实例变量都为一个属性。因此你可以添加所谓的属性观察者。在 Swift 中有两种类型的属性观察者:一种在赋值之前调用,另一种在赋值之后调用。
在赋值后调用的属性观察者采用 didSet
关键字标记。在我们的示例中,你可以使用它来监测新设置的值:
class Circle { var radius: Double = 0 { didSet { if radius < 0 { radius = oldValue } } } } let circle = Circle() circle.radius = -10 print("radius: \(circle.radius)") //radius: 0.0 circle.radius = 10 print("radius: \(circle.radius)") //radius: 10.0
在属性观察者中你可以通过变量 oldValue
来访问属性的旧值。
你还可以使用 willSet
属性观察者,它在赋值之前会被调用:
class Circle { var radius: Double = 0 { willSet { print("About to assign the new value \(newValue)") } didSet { if radius < 0 { radius = oldValue } } } } let circle = Circle() circle.radius = 10 //设置新值 10.0
在 willSet
中,你可以通过变量 newValue
来访问属性的新值。
计算属性
与存储属性不同的是,计算属性并不会存储属性的值。因此在每次调用计算属性时,都要计算该值。在 Circle
类中,你可以将属性 area
定义为计算属性:
class Circle { var radius: Double = 0 { didSet { if radius < 0 { radius = oldValue } } } var area: Double { get { return radius * radius * Double.pi } } } let circle = Circle() circle.radius = 5 print("area: \(circle.area)") //area: 78.5398163397448
计算属性总是需要一个 getter
。如果缺少 setter
,则该属性被称为只读属性。下面这个例子很好地说明了 setter
的作用:
import Foundation class Circle { var radius: Double = 0 { didSet { if radius < 0 { radius = oldValue } } } var area: Double { get { return radius * radius * Double.pi } set(newArea) { radius = sqrt(newArea / Double.pi) } } } let circle = Circle() circle.area = 25 print("radius: \(circle.radius)") //radius: 2.82094791773878
至此,每次对 area 设置了新的值之后,radius 都会被重新计算。
存储属性的初始化
每个存储属性在它的对象实例化之后都必须有值。属性初始化有两种方法:
init
下面的例子同时使用了这两种方法:
class Circle { var radius: Double var identifier: Int = 0 init(radius: Double) { self.radius = radius } } var circle = Circle(radius: 5)
如果存储属性在对象实例化之后没有值,代码无法通过编译。
懒加载属性
如果具有默认值的存储属性使用了关键字 lazy
标记,则其默认值不会立即初始化,而是在第一次访问该属性时初始化。
因此,如果该属性从未被访问,它将永远不会被初始化。你可以将这种特性应用于一些特别耗费 CPU 或内存的初始化上。
class TestClass { lazy var testString: String = "TestString" } let testClass = TestClass() print(testClass.testString) //TestString
该属性在被访问之前不会进行初始化。在这个例子中并不容易看出来。但由于初始化也可以在 block 里面实现,我们可以使它更明显一些:
class TestClass { lazy var testString: String = { print("about to initialize the property") return "TestString" }() } let testClass = TestClass() print("before first call") print(testClass.testString) print(testClass.testString)
这个例子的输出:
before first call about to initialize the property TestString TestString
这意味着该 block 仅被调用一次 - 第一次访问该属性的时候。由于存储属性是可变的,因此可以更改初始值。
类型属性
类型属性是类的一部分,但不是实例的一部分,类型属性也被称为静态属性。存储属性和计算属性都可以是类型属性。类型属性的关键字是 static
:
class TestClass { static var testString: String = "TestString" } print("\(TestClass.testString)") //TestString
如你所见,它们使用类名而不是实例对象来访问它们。此外,由于类型属性没有初始化方法,它总是需要一个默认值。
拥有私有 Setter 的公共属性
正如我在 另一篇文章 中介绍的那样,这是一种常见的情况,你不想提供一个公共的 setter,而是提供一个私有的 setter。这是封装的基本原则。这样只有类本身可以操作该属性,但仍可从类外部访问读取它。
来看下面的例子:
public class Circle { public private(set) var area: Double = 0 public private(set) var diameter: Double = 0 public var radius: Double { didSet { calculateFigures() } } public init(radius:Double) { self.radius = radius calculateFigures() } private func calculateFigures() { area = Double.pi * radius * radius diameter = 2 * Double.pi * radius } } let circle = Circle(radius: 5) print("area: \(circle.area)") //area: 78.5398163397448 print("diameter: \(circle.diameter)") //diameter: 31.4159265358979 circle.area = 10 //编译错误:无法对 'area' 属性进行赋值,因为 setter 方法不可访问
这里的属性 area
和 diameter
可以从类的外部访问,但只能在类内部赋值。为此你必须使用 public private(set)
的组合。根据本人的经验,这个特性在 iOS 开发中很少使用,但它对写出更少 bug 的代码很有帮助。
本文由 SwiftGG 翻译组翻译,已经获得作者翻译授权,最新文章请访问http://swift.gg。
以上就是本文的全部内容,希望本文的内容对大家的学习或者工作能带来一定的帮助,也希望大家多多支持 码农网
猜你喜欢:- CSS 属性篇(七):Display属性
- JavaScript对象之数据属性与访问器属性
- Logback file属性 与 fileNamePattern属性的关系
- 浅谈Spring Boot 属性配置和自定义属性配置
- python – Django属性错误. ‘module’对象没有属性’rindex’
- 深度解析属性动画的思想 - 带你手动实现属性动画框架
本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们。
Java高并发编程详解
汪文君 / 机械工业出版社 / 2018-6 / 89.00元
本书共分为四个部分:部分详细地介绍了Java多线程的基本用法和各个API的使用,并且着重介绍了线程与Java虚拟机内存之间的关系。第二部分由线程上下文类加载器方法引入,介绍为什么在线程中要有上下文类加载器的方法函数,从而掌握类在JVM的加载和初始化的整个过程。第三部分主要围绕着volatile关键字展开,在该部分中我们将会了解到现代CPU的架构以及Java的内存模型(JMM)。后一部分,主要站在架......一起来看看 《Java高并发编程详解》 这本书的介绍吧!