C票从、C++ 和 大部分更早期的語言都遵循了指令式編程的范式即支持三種語句
- 運(yùn)算語句
- 條件語句
- 循環(huán)語句
通過這些語句的組合逐一指示計(jì)算機(jī)如何工作。指令的設(shè)計(jì)貼近計(jì)算機(jī)的運(yùn)算硬件設(shè)計(jì)柠新,讓其高效運(yùn)行。雖然有利于編譯器的編寫和最終的運(yùn)行效率辉巡,但是卻阻礙了復(fù)雜程序的設(shè)計(jì)恨憎。
“舉個(gè)簡單的例子,比如我們有一個(gè)學(xué)生系統(tǒng)郊楣,記錄了學(xué)生姓名憔恳,并用一個(gè)字典記錄了各科目的考試成績:
struct Student {
let name: String
let scroes: [Subject: Int]
}
enum Subject: String, CaseIterable {
case Chinese, Mathematics, English, Physics
}
// 定義一些學(xué)生數(shù)據(jù)
let stu01 = Student(name: "stu01", scroes: [Subject.Chinese: 87, Subject.Mathematics: 82, Subject.English: 90, Subject.Physics: 86])
let stu02 = Student(name: "stu02", scroes: [Subject.Chinese: 77, Subject.Mathematics: 72, Subject.English: 80, Subject.Physics: 76])
let stu03 = Student(name: "stu03", scroes: [Subject.Chinese: 97, Subject.Mathematics: 92, Subject.English: 90, Subject.Physics: 96])
// 班級(jí)所有學(xué)生集合
let students = [stu01, stu02, stu03]
此時(shí)要計(jì)算同學(xué)的平均分,并且輸出第一名的姓名
使用指令式編程:
var best: (Student, Double)?
for s in students {
var totalScore = 0
for key in Subject.allCases {
totalScore += s.scroes[key] ?? 0
}
let averageScore = Double(totalScore) / Double(Subject.allCases.count)
if let temp = best {
if averageScore > temp.1 {
best = (s, averageScore)
}
} else {
best = (s, averageScore)
}
}
if let best = best {
print("最高平均分: \(best.1), 姓名: \(best.0.name)")
} else {
print("students 為空")
}
// 輸出
// 最高平均分: 93.75, 姓名: stu03
如果是第一次閱讀這段代碼净蚤,想要知道最終會(huì)得到什么樣的結(jié)果钥组,你可能需要仔細(xì)閱讀并理解每一行指令。但是隨著代碼量增大今瀑,維護(hù)難度加大程梦,產(chǎn)生 bug
的可能性也會(huì)成比例增長点把。
如果使用聲明式編程會(huì)是什么樣子的呢?
聲明式編程范式站在了指令式的對(duì)立面屿附,重在描述結(jié)果郎逃,讓計(jì)算機(jī)為我們考慮和組織出過程,最后得到被描述的結(jié)果拿撩。
func average(_ scores: [Subject: Int]) -> Double {
return Double(scores.values.reduce(0, +)) /
Double(Subject.allCases.count)
}
let bestStudent = students
.map { ($0, average($0.scroes)) }
.sorted { $0.1 > $1.1 }
.first
print("最高平均分:\(bestStudent?.1 ?? 0.0)" + "衣厘,姓名:\(bestStudent?.0.name ?? "")")
// 輸出
// 最高平均分:93.75,姓名:stu03
我們首先將 students
映射為一個(gè) (Student, 平均分)
的數(shù)組压恒,然后按照平均分進(jìn)行排序影暴,最后取出我們想要的結(jié)果。
部分內(nèi)容摘抄自 SwiftUI 與 Combine 編程