Scala开发教程(43): 组合和继承–小结

jerry Scala 2015年11月25日 收藏

前面我们基本完成了布局元素的函数库,现在我们就可以写个程序来使用这个函数库,下面显示螺旋线的程序如下:

  1. object Spiral {
  2. val space = elem (" ")
  3. val corner = elem ("+")
  4. def spiral(nEdges:Int, direction:Int): Element = {
  5. if(nEdges==1)
  6. elem("+")
  7. else{
  8. val sp=spiral(nEdges -1, (direction +3) % 4)
  9. def verticalBar = elem ('|',1, sp.height)
  10. def horizontalBar = elem('-',sp.width,1)
  11. if(direction==0)
  12. (corner beside horizontalBar) above (sp beside space)
  13. else if (direction ==1)
  14. (sp above space) beside ( corner above verticalBar)
  15. else if(direction ==2 )
  16. (space beside sp) above (horizontalBar beside corner)
  17. else
  18. (verticalBar above corner) beside (space above sp)
  19. }
  20. }
  21. def main(args:Array[String]) {
  22. val nSides=args(0).toInt
  23. println(spiral(nSides,0))
  24. }
  25. }

因为Sprial为一单例对象,并包含main方法,因此它为一Scala应用程序,可以在命令行使用scala Sprial xx来运行这个应用。

  1. root@mail:~/scala# scala Spiral 5
  2. +----
  3. |
  4. | ++
  5. | |
  6. +--+
  7.  
  8. root@mail:~/scala# scala Spiral 23
  9. +----------------------
  10. |
  11. | +------------------+
  12. | | |
  13. | | +--------------+ |
  14. | | | | |
  15. | | | +----------+ | |
  16. | | | | | | |
  17. | | | | +------+ | | |
  18. | | | | | | | | |
  19. | | | | | +--+ | | | |
  20. | | | | | | | | | | |
  21. | | | | | ++ | | | | |
  22. | | | | | | | | | |
  23. | | | | +----+ | | | |
  24. | | | | | | | |
  25. | | | +--------+ | | |
  26. | | | | | |
  27. | | +------------+ | |
  28. | | | |
  29. | +----------------+ |
  30. | |
  31. +--------------------+

这个例子的完整代码如下:

  1. object Element {
  2. private class ArrayElement(val contents: Array[String])
  3. extends Element
  4.  
  5. private class LineElement(s:String) extends Element {
  6. val contents=Array(s)
  7. override def width = s.length
  8. override def height = 1
  9. }
  10.  
  11. private class UniformElement (ch :Char,
  12. override val width:Int,
  13. override val height:Int
  14. ) extends Element{
  15. private val line=ch.toString * width
  16. def contents = Array.fill(height)(line)
  17. }
  18.  
  19. def elem(contents: Array[String]):Element =
  20. new ArrayElement(contents)
  21.  
  22. def elem(chr:Char, width:Int, height:Int) :Element =
  23. new UniformElement(chr,width,height)
  24.  
  25. def elem(line:String) :Element =
  26. new LineElement(line)
  27. }
  28.  
  29. import Element.elem
  30.  
  31. abstract class Element {
  32. def contents: Array[String]
  33. def height: Int = contents.length
  34. def width: Int = contents(0).length
  35. def above(that: Element) :Element = {
  36. val this1=this widen that.width
  37. val that1=that widen this.width
  38. elem (this1.contents ++ that1.contents)
  39. }
  40.  
  41.  
  42. def beside(that: Element) :Element = {
  43. val this1=this heighten that.height
  44. val that1=that heighten this.height
  45. Element.elem(
  46. for(
  47. (line1,line2) <- this1.contents zip that1.contents
  48. ) yield line1+line2
  49. )
  50. }
  51.  
  52. def widen(w: Int): Element =
  53. if (w <= width) this
  54. else {
  55. val left = Element.elem(' ', (w - width) / 2, height)
  56. var right = Element.elem(' ', w - width - left.width, height)
  57. left beside this beside right
  58. }
  59.  
  60. def heighten(h: Int): Element =
  61. if (h <= height) this
  62. else {
  63. val top = Element.elem(' ', width, (h - height) / 2)
  64. var bot = Element.elem(' ', width, h - height - top.height)
  65. top above this above bot
  66. }
  67. override def toString = contents mkString "\n"
  68.  
  69. }
  70.  
  71. object Spiral {
  72. val space = elem (" ")
  73. val corner = elem ("+")
  74. def spiral(nEdges:Int, direction:Int): Element = {
  75. if(nEdges==1)
  76. elem("+")
  77. else{
  78. val sp=spiral(nEdges -1, (direction +3) % 4)
  79. def verticalBar = elem ('|',1, sp.height)
  80. def horizontalBar = elem('-',sp.width,1)
  81. if(direction==0)
  82. (corner beside horizontalBar) above (sp beside space)
  83. else if (direction ==1)
  84. (sp above space) beside ( corner above verticalBar)
  85. else if(direction ==2 )
  86. (space beside sp) above (horizontalBar beside corner)
  87. else
  88. (verticalBar above corner) beside (space above sp)
  89. }
  90. }
  91.  
  92. def main(args:Array[String]) {
  93. val nSides=args(0).toInt
  94. println(spiral(nSides,0))
  95. }
  96. }
  97.  

到目前为止,你看到了Scala里与面向对象编程有关的更多的概念。其中,你遇到了抽象类,继承和派生,类层次,参数化字段,及方法重载。你应当已经建立了在Scala里构造不太小的类层次的感觉。