这篇文章将介绍类的基础知识
定义
Scala 中以 class 来作为类的声明,在类中可以定义成员和方法,成员和方法可以有不同的可见性(这个会在后文详述)。
scala> class Company {
| private var employeeCount = 0
| def getEmployeeCount(): Int = employeeCount
| def setEmployeeCount( count: Int)= {
| employeeCount = count
| }
|
| def m( i: Int ) {}
| def m( str: String ) {}
| }
defined class Company
构造器
Scala 中,类有一个主构造器,主构造器必须包含所需的所有参数。除了一个主构造器,还可以有0个或多个辅助构造器,辅助构造器又称次构造器。辅助构造器命名为 this,其第一条语句必须调用主构造器或其他辅助构造器,来看下面的例子:
scala> class T ( x1: Int, y1: String, z1: Double ) {
| private val xx1 = x1
| private val yy1 = y1
| private val zz1 = z1
|
| def this ( x1: Int, y1: String ) {
| this( x1, y1, 1.0 )
| }
|
| def this ( x1: Int ) {
| this( x1, "" )
| }
| }
defined class T
还有一点需要注意的是,被调用的辅助构造函数的定义必须放在主动调用的辅助构造函数前面,不然会报错:
scala> class T ( x1: Int, y1: String, z1: Double ) {
| private val xx1 = x1
| private val yy1 = y1
| private val zz1 = z1
|
| def this ( x1: Int ) {
| this( x1, "" )
| }
|
| def this ( x1: Int, y1: String ) {
| this( x1, y1, 1.0 )
| }
| }
<console>:13: error: called constructor's definition must precede calling constructor's definition
this( x1, "" )
^
不管辅助函数调来调去,最终都还是要调用到主构造函数,这确保了新实例的初始化逻辑一致。
如果在主构造函数的参数前加 var 或 val,该参数就成为实例的一个成员,这部分知识在Scala case class那些你不知道的知识有更详细的介绍
重载
Scala 类方法允许重载,如类 Company 中的 m 方法。重载要求参数列表和返回类型不完全相同,但参数名可相同,这是因为编译后是通过方法名、参数列表、返回类型综合来区分各个方法的。
在方法重载时,有一点需要注意:对于『高级类型』,存在类型擦除机制,所谓的高级类型就是包含类型参数的类型,比如 List[A],下面这个例子可以展示了类型擦除:
scala> class Tmp {
| def m( data: List[Int] ) {}
| def m( data: List[String] ) {}
| }
<console>:9: error: double definition:
method m:(data: List[String])Unit and
method m:(data: List[Int])Unit at line 8
have same type after erasure: (data: List)Unit
def m( data: List[String] ) {}
^
报了有相同类型的参数的错误。
类型成员
Scala 允许你在类内部定义类型成员,在构造类实例的时候指定该类型成员对应的具体类型。类型成员可用于类内部的成员或函数,提供了更好的泛华能力,从下面这个简单的例子可以看出:
scala> class T {
| type X
|
| def getClassName( x: X): String = {
| x.getClass.getTypeName
| }
| }
defined class T
scala> val x1 = new T{ type X = Int }
x1: T{type X = Int} = $anon$1@515f550a
scala> x1.getClassName(10)
res0: String = java.lang.Integer
scala> val x2 = new T{ type X = String }
x2: T{type X = String} = $anon$1@61a52fbd
scala> x2.getClassName("string")
res1: String = java.lang.String
当然,也可以在类外部定义类型变量,如:
scala> type L = List[Int]
defined type alias L
方法与成员同名
与 JAVA 不同,如果方法参数列表不为空,该方法可以与成员同名,如:
scala> class T {
| private val m = 0
|
| def m( i: Int ): Int = m + i
| }
defined class T