Scala 類(lèi)和對(duì)象
類(lèi)是對(duì)象的抽象,而對(duì)象是類(lèi)的具體實(shí)例。類(lèi)是抽象的,不占用內(nèi)存,而對(duì)象是具體的,占用存儲(chǔ)空間。類(lèi)是用于創(chuàng)建對(duì)象的藍(lán)圖,它是一個(gè)定義包括在特定類(lèi)型的對(duì)象中的方法和變量的軟件模板。
我們可以使用 new 關(guān)鍵字來(lái)創(chuàng)建類(lèi)的對(duì)象,實(shí)例如下:
class Point(xc: Int, yc: Int) { var x: Int = xc var y: Int = yc def move(dx: Int, dy: Int) { x = x + dx y = y + dy println ("x 的坐標(biāo)點(diǎn): " + x); println ("y 的坐標(biāo)點(diǎn): " + y); } }
Scala中的類(lèi)不聲明為public,一個(gè)Scala源文件中可以有多個(gè)類(lèi)。
以上實(shí)例的類(lèi)定義了兩個(gè)變量 x 和 y ,一個(gè)方法:move,方法沒(méi)有返回值。
Scala 的類(lèi)定義可以有參數(shù),稱(chēng)為類(lèi)參數(shù),如上面的 xc, yc,類(lèi)參數(shù)在整個(gè)類(lèi)中都可以訪問(wèn)。
接著我們可以使用 new 來(lái)實(shí)例化類(lèi),并訪問(wèn)類(lèi)中的方法和變量:
import java.io._ class Point(xc: Int, yc: Int) { var x: Int = xc var y: Int = yc def move(dx: Int, dy: Int) { x = x + dx y = y + dy println ("x 的坐標(biāo)點(diǎn): " + x); println ("y 的坐標(biāo)點(diǎn): " + y); } } object Test { def main(args: Array[String]) { val pt = new Point(10, 20); // 移到一個(gè)新的位置 pt.move(10, 10); } }
執(zhí)行以上代碼,輸出結(jié)果為:
$ scalac Test.scala $ scala Test x 的坐標(biāo)點(diǎn): 20 y 的坐標(biāo)點(diǎn): 30
Scala 繼承
Scala繼承一個(gè)基類(lèi)跟Java很相似, 但我們需要注意一下幾點(diǎn):
1、重寫(xiě)一個(gè)非抽象方法必須使用override修飾符。
2、只有主構(gòu)造函數(shù)才可以往基類(lèi)的構(gòu)造函數(shù)里寫(xiě)參數(shù)。
3、在子類(lèi)中重寫(xiě)超類(lèi)的抽象方法時(shí),你不需要使用override關(guān)鍵字。
接下來(lái)讓我們來(lái)看個(gè)實(shí)例:
class Point(xc: Int, yc: Int) { var x: Int = xc var y: Int = yc def move(dx: Int, dy: Int) { x = x + dx y = y + dy println ("x 的坐標(biāo)點(diǎn): " + x); println ("y 的坐標(biāo)點(diǎn): " + y); } } class Location(override val xc: Int, override val yc: Int, val zc :Int) extends Point(xc, yc){ var z: Int = zc def move(dx: Int, dy: Int, dz: Int) { x = x + dx y = y + dy z = z + dz println ("x 的坐標(biāo)點(diǎn) : " + x); println ("y 的坐標(biāo)點(diǎn) : " + y); println ("z 的坐標(biāo)點(diǎn) : " + z); } }
Scala 使用 extends 關(guān)鍵字來(lái)繼承一個(gè)類(lèi)。實(shí)例中 Location 類(lèi)繼承了 Point 類(lèi)。Point 稱(chēng)為父類(lèi)(基類(lèi)),Location 稱(chēng)為子類(lèi)。
override val xc 為重寫(xiě)了父類(lèi)的字段。
繼承會(huì)繼承父類(lèi)的所有屬性和方法,Scala 只允許繼承一個(gè)父類(lèi)。
實(shí)例如下:
import java.io._ class Point(val xc: Int, val yc: Int) { var x: Int = xc var y: Int = yc def move(dx: Int, dy: Int) { x = x + dx y = y + dy println ("x 的坐標(biāo)點(diǎn) : " + x); println ("y 的坐標(biāo)點(diǎn) : " + y); } } class Location(override val xc: Int, override val yc: Int, val zc :Int) extends Point(xc, yc){ var z: Int = zc def move(dx: Int, dy: Int, dz: Int) { x = x + dx y = y + dy z = z + dz println ("x 的坐標(biāo)點(diǎn) : " + x); println ("y 的坐標(biāo)點(diǎn) : " + y); println ("z 的坐標(biāo)點(diǎn) : " + z); } } object Test { def main(args: Array[String]) { val loc = new Location(10, 20, 15); // 移到一個(gè)新的位置 loc.move(10, 10, 5); } }
執(zhí)行以上代碼,輸出結(jié)果為:
$ scalac Test.scala $ scala Test x 的坐標(biāo)點(diǎn) : 20 y 的坐標(biāo)點(diǎn) : 30 z 的坐標(biāo)點(diǎn) : 20
Scala重寫(xiě)一個(gè)非抽象方法,必須用override修飾符。
class Person { var name = "" override def toString = getClass.getName + "[name=" + name + "]" } class Employee extends Person { var salary = 0.0 override def toString = super.toString + "[salary=" + salary + "]" } object Test extends App { val fred = new Employee fred.name = "Fred" fred.salary = 50000 println(fred) }
執(zhí)行以上代碼,輸出結(jié)果為:
$ scalac Test.scala $ scala Test Employee[name=Fred][salary=50000.0]
Scala 單例對(duì)象
在 Scala 中,是沒(méi)有 static 這個(gè)東西的,但是它也為我們提供了單例模式的實(shí)現(xiàn)方法,那就是使用關(guān)鍵字 object。
Scala 中使用單例模式時(shí),除了定義的類(lèi)之外,還要定義一個(gè)同名的 object 對(duì)象,它和類(lèi)的區(qū)別是,object對(duì)象不能帶參數(shù)。
當(dāng)單例對(duì)象與某個(gè)類(lèi)共享同一個(gè)名稱(chēng)時(shí),他被稱(chēng)作是這個(gè)類(lèi)的伴生對(duì)象:companion object。你必須在同一個(gè)源文件里定義類(lèi)和它的伴生對(duì)象。類(lèi)被稱(chēng)為是這個(gè)單例對(duì)象的伴生類(lèi):companion class。類(lèi)和它的伴生對(duì)象可以互相訪問(wèn)其私有成員。
單例對(duì)象實(shí)例
import java.io._ class Point(val xc: Int, val yc: Int) { var x: Int = xc var y: Int = yc def move(dx: Int, dy: Int) { x = x + dx y = y + dy } } object Test { def main(args: Array[String]) { val point = new Point(10, 20) printPoint def printPoint{ println ("x 的坐標(biāo)點(diǎn) : " + point.x); println ("y 的坐標(biāo)點(diǎn) : " + point.y); } } }
執(zhí)行以上代碼,輸出結(jié)果為:
$ scalac Test.scala $ scala Test x 的坐標(biāo)點(diǎn) : 10 y 的坐標(biāo)點(diǎn) : 20
伴生對(duì)象實(shí)例
/* 文件名:Marker.scala * author:php中文網(wǎng) * url:m.miracleart.cn */ // 私有構(gòu)造方法 class Marker private(val color:String) { println("創(chuàng)建" + this) override def toString(): String = "顏色標(biāo)記:"+ color } // 伴生對(duì)象,與類(lèi)共享名字,可以訪問(wèn)類(lèi)的私有屬性和方法 object Marker{ private val markers: Map[String, Marker] = Map( "red" -> new Marker("red"), "blue" -> new Marker("blue"), "green" -> new Marker("green") ) def apply(color:String) = { if(markers.contains(color)) markers(color) else null } def getMarker(color:String) = { if(markers.contains(color)) markers(color) else null } def main(args: Array[String]) { println(Marker("red")) // 單例函數(shù)調(diào)用,省略了.(點(diǎn))符號(hào) println(Marker getMarker "blue") } }
執(zhí)行以上代碼,輸出結(jié)果為:
$ scalac Marker.scala $ scala Marker 創(chuàng)建顏色標(biāo)記:red 創(chuàng)建顏色標(biāo)記:blue 創(chuàng)建顏色標(biāo)記:green 顏色標(biāo)記:red 顏色標(biāo)記:blue