继续领悟函数式:Scala指令式风格代码的重构

为了帮助你在函数式风格上获得更多的领悟,本节我们将重构代码7.18中以指令式风格打印乘法表的方式。我们的函数式替代品展示在代码7.19中。

编辑推荐:Scala编程语言专题

代码7.18中的代码在两个方面显示出了指令式风格。首先,调用printMultiTable有副作用:在标准输出上打印乘法表。代码7.19中,我们重构了函数,让它把乘法表作为字串返回。由于函数不再执行打印,我们把它重命名为multiTable。正如前面提到过的,没有副作用的函数的一个优点是它们很容易进行单元测试。要测试printMultiTable,你需要重定义print和println从而能够检查输出的正确性。测试multiTable就简单多了,只要检查结果即可。

 
 
 
  1. // 以序列形式返回一行乘法表  
  2. def makeRowSeq(row: Int) =  
  3.  for (col < - 1 to 10) yield {  
  4.   val prod = (row * col).toString  
  5.   val padding = " " * (4 - prod.length)  
  6.   padding + prod  
  7.  }  
  8. // 以字串形式返回一行乘法表  
  9. def makeRow(row: Int) = makeRowSeq(row).mkString  
  10. // 以字串形式返回乘法表,每行记录占一行字串  
  11. def multiTable() = {  
  12.  val tableSeq = // 行记录字串的序列  
  13.   for (row < - 1 to 10)  
  14.    yield makeRow(row)  
  15.  tableSeq.mkString("\n")  
  16. }  

代码 7.19 创建乘法表的函数式方法

printMultiTable里另一个揭露其指令式风格的信号来自于它的while循环和var。与之相对,multiTable函数使用了val,for表达式,帮助函数:helper function,并调用了mkString。

我们提炼出两个帮助函数,makeRow和makeRowSeq,使代码容易阅读。函数makeRowSeq使用for表达式从1到10枚举列数。这个for函数体计算行和列的乘积,决定乘积前占位的空格,并生成由占位空格,乘积字串叠加成的结果。for表达式的结果是一个包含了这些生成字串作为元素的序列(scala.Seq的某个子类)。另一个帮助函数,makeRow,仅仅调用了makeRowSeq返回结果的mkString函数。叠加序列中的字串把它们作为一个字串返回。

multiTable方法首先使用一个for表达式的结果初始化tableSeq,这个for表达式从1到10枚举行数,对每行调用makeRow获得该行的字串。因为字串前缀yield关键字,所以表达式的结果就是行字串的序列。现在仅剩下的工作就是把字串序列转变为单一字串。mkString的调用完成这个工作,并且由于我们传递进去"\n",因此每个字串结尾插入了换行符。如果把multiTable返回的字串传递给println,你将看到与调用printMultiTable所生成的同样的输出结果。

【相关阅读】

  1. 学习Scala的变量范围
  2. Scala:match表达式、break和continue
  3. 学习Scala:使用try-catch表达式处理异常
  4. Scala中的for表达式:枚举的“瑞士军刀”
  5. Scala中的if表达式和while循环
THE END