扩展类

scala扩展类的方式和Java一样,使用extends关键字:

1
2
3
4
class Employee extends Person{
val salary = 0.0
...
}

和Java一样,你可以将类声明为final,这样它就不能被扩展。你可以将单个方法或者字段声明为final, 以确保它们不能够被重写。

重写方法

在scala中重写一个非抽象方法必须使用override修饰符。

1
2
3
4
public class Person{
...
override def toString = getClass.getName + "[name=" + name + "]"
}

override修饰符可以用在多个情况下给出有用的错误提示。包括:

  • 当你拼错要重写的方法名
  • 当你不小心在新方法中使用错误的参数类型。
  • 当你在超类中引入新的方法,而这个新的方法与子类的方法相抵触。

在scala中调用超类的方法和Java一样,使用super关键字。

类型转换和检查

要检查某个对象是否属于某个特定的类,可以用isInstanceOf方法。如果测试成功,你就可以使用asInstanceOf方法将引用转换为子类的引用。

1
2
3
if (p.isInstanceOf[Employee]){
val s = p.asInstanceOf[Employee]
}

如果p指向的是Employee类及其子类,则p.isInstanceOf[Employee]就会成功;

如果p是null, 则p.isInstanceOf[Employee]将返回false,且p.asInstanceOf[Employee]将返回null。

如果p不是一个Employee,则p.asInstanceOf[Employee]将抛出异常。

不过与类型检查和转化相比,模式匹配通常是更好的选择。

受保护字段

如果字段或方法被声明为protected,则这样的成员可以被任何子类访问,但不能从其他位置看到。与Java不同的是,protected的成员对于类所属的包而言,是不可见的。

超类的构造

辅助构造器永远不能够直接调用超类的构造器,子类的辅助构造器最终都会调用主构造器,只有主构造器可以调用超类的构造器。

scala类可以扩展Java类,这种情况下,它的主构造器必须调用Java超类的某一个构造方法。

重写字段

注意以下限制:

  • def只能重写另一个def
  • val只能重写另一个val或不带参数的def
  • var只能重写另一个抽象的var

匿名子类

和Java一样,你可以通过包含带有定义或者重写代码块的方式创建一个匿名子类。

1
2
3
val align = new Person("Fred"){
def greeting = "hello"
}

抽象类

和Java一样,可以使用abstract关键字定义一个不能被实例化的类。

1
2
3
abstract class Person(val name:String){
def id:Int
}

但是在scala中,不像java,你不需要对抽象方法使用abstract关键字,你只是省去方法体。

在子类中重写父类的抽象方法时,你不需要使用override关键字。

抽象字段

除了抽象方法之外,类还可以拥有抽象字段,抽象字段就是一个没有初始化值的字段。

具体的子类必须提供具体的字段。和方法一样,在子类中重写超类的抽象字段时,不需要override关键字。

scala继承层级

所有其他类都是AnyRef的子类,AnyRef相当于Java中的Object类。

AnyVal和AnyRef都扩展自Any类,而Any类是整个继承层级的根节点。

Null类型的唯一实例就是null值,你可以额将null值赋值给任何引用,但不能赋值给值类型的应用。

Nothing类型没有实例,它对于泛型结构时常有用。

对象相等性

在scala中,AnyRef的eq方法检查两个引用是否指向同一个对象。AnyRef的equals方法调用eq。