私有访问修饰符限制它所定义的实体以及实体中的任何类型的访问——也称为“词汇范围”。同一源文件中类型的扩展也可以访问实体的。
为了证明这一点,可以通过扩展BasicAccount创建一个CheckingAccount(支票账户)。
class CheckingAccount: BasicAccount {
private let accountNumber = UUID().uuidString
class Check {
let account: String
var amount: Dollars
//兑现
private(set) var cashed = false
func cash() {
cashed = true
}
init(amount: Dollars, from account: CheckingAccount) {
self.amount = amount
self.account = account.accountNumber
}
}
}
CheckingAccount有一个声明为private的accountNumber。CheckingAccount还有一个嵌套类Check,可以在其初始化方法中读取私有值accountNumber。
注意:在本例中,UUID类用于生成唯一的帐户号。这个类是Foundation的一部分,所以不要忘记导入它!
CheckingAccount能够开支票和兑换现金。在CheckingAccount中添加以下方法:
func writeCheck(amount: Dollars) -> Check? {
guard balance > amount else {
return nil
}
let check = Check(amount: amount, from: self)
withdraw(amount: check.amount)
return check
}
func deposit(_ check: Check) {
guard !check.cashed else {
return
}
deposit(amount: check.amount)
check.cash()
}
虽然CheckingAccount仍然可以进行基本的存款和取款,但是它现在的 writeCheck(amount:) 在提取金额和创建支票之前需要检查是否有足够的余额,并且如果支票已经兑现,那么deposit(_:)方法将不会存款。
让约翰给简写张支票,在你的playground上试一试这个代码。
// 创建一个John的支票账户存入 $300.00
let johnChecking = CheckingAccount()
johnChecking.deposit(amount: 300.00)
// 开一个 $200.00的支票
let check = johnChecking.writeCheck(amount: 200.0)!
// 创建一个Jane的支票账户, 把 john的支票存进去
let janeChecking = CheckingAccount()
janeChecking.deposit(check)
janeChecking.balance // 200.00
// 再次把支票存入,没有影响。
// Jane's balance this time :]
janeChecking.deposit(check)
janeChecking.balance // 200.00
使用点语法查看哪些是CheckingAccount的接口。
QQ20180717-094616@2x.pngaccountNumber被视为CheckingAccount的实现细节,对于使用代码是不可见的。
同时,Check使cashed私有了,所以使用cash()方法代替
QQ20180717-095207@2x.png这个接口提供了一种方法,可以将支票标记为已经兑换,但不能反过来。换句话说,不可能取消兑现支票。
最后,尽管在CheckingAccount无法看到accountNumber,但是任何持有支票的人都可以访问这个号码:
QQ20180717-095917@2x.png
重要的是,不管这个代码是如何实现的,访问修饰符让代码形成自己的接口。
网友评论