繼續(xù)領(lǐng)悟函數(shù)式:Scala指令式風(fēng)格代碼的重構(gòu)
為了幫助你在函數(shù)式風(fēng)格上獲得更多的領(lǐng)悟,本節(jié)我們將重構(gòu)代碼7.18中以指令式風(fēng)格打印乘法表的方式。我們的函數(shù)式替代品展示在代碼7.19中。
51CTO編輯推薦:Scala編程語言專題
代碼7.18中的代碼在兩個方面顯示出了指令式風(fēng)格。首先,調(diào)用printMultiTable有副作用:在標(biāo)準(zhǔn)輸出上打印乘法表。代碼7.19中,我們重構(gòu)了函數(shù),讓它把乘法表作為字串返回。由于函數(shù)不再執(zhí)行打印,我們把它重命名為multiTable。正如前面提到過的,沒有副作用的函數(shù)的一個優(yōu)點(diǎn)是它們很容易進(jìn)行單元測試。要測試printMultiTable,你需要重定義print和println從而能夠檢查輸出的正確性。測試multiTable就簡單多了,只要檢查結(jié)果即可。
代碼 7.19 創(chuàng)建乘法表的函數(shù)式方法
- // 以序列形式返回一行乘法表
- def makeRowSeq(row: Int) =
- for (col < - 1 to 10) yield {
- val prod = (row * col).toString
- val padding = " " * (4 - prod.length)
- padding + prod
- }
- // 以字串形式返回一行乘法表
- def makeRow(row: Int) = makeRowSeq(row).mkString
- // 以字串形式返回乘法表,每行記錄占一行字串
- def multiTable() = {
- val tableSeq = // 行記錄字串的序列
- for (row < - 1 to 10)
- yield makeRow(row)
- tableSeq.mkString("\n")
- }
printMultiTable里另一個揭露其指令式風(fēng)格的信號來自于它的while循環(huán)和var。與之相對,multiTable函數(shù)使用了val,for表達(dá)式,幫助函數(shù):helper function,并調(diào)用了mkString。
我們提煉出兩個幫助函數(shù),makeRow和makeRowSeq,使代碼容易閱讀。函數(shù)makeRowSeq使用for表達(dá)式從1到10枚舉列數(shù)。這個for函數(shù)體計算行和列的乘積,決定乘積前占位的空格,并生成由占位空格,乘積字串疊加成的結(jié)果。for表達(dá)式的結(jié)果是一個包含了這些生成字串作為元素的序列(scala.Seq的某個子類)。另一個幫助函數(shù),makeRow,僅僅調(diào)用了makeRowSeq返回結(jié)果的mkString函數(shù)。疊加序列中的字串把它們作為一個字串返回。
multiTable方法首先使用一個for表達(dá)式的結(jié)果初始化tableSeq,這個for表達(dá)式從1到10枚舉行數(shù),對每行調(diào)用makeRow獲得該行的字串。因?yàn)樽执熬Yyield關(guān)鍵字,所以表達(dá)式的結(jié)果就是行字串的序列?,F(xiàn)在僅剩下的工作就是把字串序列轉(zhuǎn)變?yōu)閱我蛔执kString的調(diào)用完成這個工作,并且由于我們傳遞進(jìn)去"\n",因此每個字串結(jié)尾插入了換行符。如果把multiTable返回的字串傳遞給println,你將看到與調(diào)用printMultiTable所生成的同樣的輸出結(jié)果。
【相關(guān)閱讀】



















