鍍金池/ 教程/ Scala/ Scala 課堂:基礎(chǔ)(三)
Scala 課堂:高級類型(二)
Scala 課堂:基礎(chǔ)(三)
Scala 課堂:類型和多態(tài)類型(一)
Scala 課堂:基礎(chǔ)(二)
Scala課堂:基礎(chǔ)(五)
Scala 課堂:基礎(chǔ)(四)
Scala 課堂:基礎(chǔ)(六)
Scala 課堂:集合(二)
Scala課堂:類型和多態(tài)類型(二)
Scala 課堂:模式匹配和函數(shù)組合
Scala 課堂:高級類型(一)
Scala 課堂:集合(一)
Scala 課堂:基礎(chǔ)(一)

Scala 課堂:基礎(chǔ)(三)

這里我們轉(zhuǎn)載 Twitter 的 Scala 課堂 ,轉(zhuǎn)載的內(nèi)容基本來自 Twitter 的 Scala 課堂中文翻譯,部分有小改動.


scala> class Calculator {
     |         val brand: String = "HP"
     |         def add(m: Int, n: Int): Int = m + n
     |       }
defined class Calculator

scala> val calc = new Calculator
calc: Calculator = Calculator@6fe377b0

scala>  calc.add(1, 2)
res12: Int = 3

scala>  calc.brand
res13: String = HP

上面的例子展示了如何在類中用 def 定義方法和用 val 定義字段值。方法就是可以訪問類的狀態(tài)的函數(shù)。

構(gòu)造函數(shù)

構(gòu)造函數(shù)不是特殊的方法,他們是除了類的方法定義之外的代碼。讓我們擴展計算器的例子,增加一個構(gòu)造函數(shù)參數(shù),并用它來初始化內(nèi)部狀態(tài)。


class Calculator(brand: String) {
  /**
   * A constructor.
   */
  val color: String = if (brand == "TI") {
    "blue"
  } else if (brand == "HP") {
    "black"
  } else {
    "white"
  }

  // An instance method.
  def add(m: Int, n: Int): Int = m + n
}

注意兩種不同風格的評論。

你可以使用構(gòu)造函數(shù)來構(gòu)造一個實例:


scala>  val calc = new Calculator("HP")
calc: Calculator = Calculator@39c8d5c4

scala>  calc.color
res14: String = black

表達式

上文的 Calculator 例子說明了 Scala 是如何面向表達式的。顏色的值就是綁定在一個 if/else 表達式上的。Scala 是高度面向表達式的:大多數(shù)東西都是表達式而非指令

旁白: 函數(shù) vs 方法

函數(shù)和方法在很大程度上是可以互換的。由于函數(shù)和方法是如此的相似,你可能都不知道你調(diào)用的東西是一個函數(shù)還是一個方法。而當真正碰到的方法和函數(shù)之間的差異的時候,你可能會感到困惑。


scala> class C {
     |         var acc = 0
     |         def minc = { acc += 1 }
     |         val finc = { () => acc += 1 }
     |       }
defined class C

scala>  val c = new C
c: C = C@7686da2

scala>  c.minc // calls c.minc()

scala>  c.finc // returns the function as a value:
res16: () => Unit = <function0>

當你可以調(diào)用一個不帶括號的“函數(shù)”,但是對另一個卻必須加上括號的時候,你可能會想哎呀,我還以為自己知道 Scala 是怎么工作的呢。也許他們有時需要括號?你可能以為自己用的是函數(shù),但實際使用的是方法。

在實踐中,即使不理解方法和函數(shù)上的區(qū)別,你也可以用 Scala 做偉大的事情。如果你是 Scala 新手,而且在讀兩者的差異解釋,你可能會跟不上。不過這并不意味著你在使用 Scala 上有麻煩。它只是意味著函數(shù)和方法之間的差異是很微妙的,只有深入語言內(nèi)部才能清楚理解它。

繼承


class ScientificCalculator(brand: String) extends Calculator(brand) {
  def log(m: Double, base: Double) = math.log(m) / math.log(base)
}

參考 Effective Scala 指出如果子類與父類實際上沒有區(qū)別,類型別名是優(yōu)于繼承的。A Tour of Scala 詳細介紹了子類化。

重載方法


class EvenMoreScientificCalculator(brand: String) extends ScientificCalculator(brand) {
  def log(m: Int): Double = log(m, math.exp(1))
}

抽象類

你可以定義一個抽象類,它定義了一些方法但沒有實現(xiàn)它們。取而代之是由擴展抽象類的子類定義這些方法。你不能創(chuàng)建抽象類的實例。


scala> abstract class Shape {
     |         def getArea():Int    // subclass should define this
     |       }
defined class Shape

scala> class Circle(r: Int) extends Shape {
     |         def getArea():Int = { r * r * 3 }
     |       }
defined class Circle

scala>  val s = new Shape
<console>:8: error: class Shape is abstract; cannot be instantiated
        val s = new Shape
                ^

scala>  val c = new Circle(2)
c: Circle = Circle@1fe4da96

特質(zhì)(Traits)

特質(zhì)是一些字段和行為的集合,可以擴展或混入(mixin)你的類中.


trait Car {
  val brand: String
}

trait Shiny {
  val shineRefraction: Int
}

class BMW extends Car {
  val brand = "BMW"
}

通過 with 關(guān)鍵字,一個類可以擴展多個特質(zhì):


class BMW extends Car with Shiny {
  val brand = "BMW"
  val shineRefraction = 12
}

參考 Effective Scala 對[特質(zhì)的觀點](http://twitter.github.io/effectivescala/#Object oriented programming-Traits)。

什么時候應該使用特質(zhì)而不是抽象類?

如果你想定義一個類似接口的類型,你可能會在特質(zhì)和抽象類之間難以取舍。這兩種形式都可以讓你定義一個類型的一些行為,并要求繼承者定義一些其他行為。一些經(jīng)驗法則:

  • 優(yōu)先使用特質(zhì)。一個類擴展多個特質(zhì)是很方便的,但卻只能擴展一個抽象類。
  • 如果你需要構(gòu)造函數(shù)參數(shù),使用抽象類。因為抽象類可以定義帶參數(shù)的構(gòu)造函數(shù),而特質(zhì)不行。例如,你不能說 trait t(i: Int) {},參數(shù) i 是非法的。

你不是問這個問題的第一人??梢圆榭锤娴拇鸢福?stackoverflow: Scala 特質(zhì) vs 抽象類 , 抽象類和特質(zhì)的區(qū)別, and Scala 編程: 用特質(zhì),還是不用特質(zhì)?

類型

此前,我們定義了一個函數(shù)的參數(shù)為 Int,表示輸入是一個數(shù)字類型。其實函數(shù)也可以是泛型的,來適用于所有類型。當這種情況發(fā)生時,你會看到用方括號語法引入的類型參數(shù)。下面的例子展示了一個使用泛型鍵和值的緩存。


trait Cache[K, V] {
  def get(key: K): V
  def put(key: K, value: V)
  def delete(key: K)
}

方法也可以引入類型參數(shù)。


def remove[K](key: K)