9、scala面向对象编程之继承
1、 extends
2、override 和super
3、override field
4、isInstanceOf和asInstanceOf
5、getClass和classOf
6、使用模式匹配进行类型判断
7、protected
8、调用父类的construct
9、匿名内部类
10、抽象类
11、抽象field
1、 extends
scala中,让子类继承父类,与java一样,也使用extends关键字。
子类可以从父类继承父类的field和method,然后子类可以在自己内部放入父类没有,子类特有的field和method,使用继承可以有效服用代码。
子类可以覆盖父类的field和method,但是如果父类用final修饰,则该类是无法被继承的; 如果父类的field和method用final修饰,则field和method是无法被覆盖的。
2、override 和super
如果子类要覆盖父类的非抽象方法,则必须许使用override关键字。
Override关键字可以帮助我们今早发现代码里的错误,比如override修饰的父类方法的方法名拼写错了,覆盖父类方法的参数我们写错了,等等。
子类覆盖父类方法后,如果我们需要调用父类被覆盖的方法,可以使用super关键字,显示的指定要调用的父类的方法
3、override field
Scala中,子类可以覆盖父类的val field,而且子类的val field还可以覆盖父类的val field的getter方法,只要在子类中使用override即可。
4、isInstanceOf和asInstanceOf
如果创建了子类的对象,但又将其赋予了子类类型的变量,后续在程序中,又需要将父类类型的变量转换为子类类型的变量。
首先,需要使用isInstanceOf判断对象是否是指定类的对象,如果是的话,则可以使用asInstanceOf将对象转换为指定类型。
如果对象是null,则isInstanceOf一定返回false,asInstanceOf一定给返回null。
如果没有isInstanceOf先判断对象是否为指定类的实例,就直接用asInstanceOf转换,则可能抛出异常。
5、getClass和classOf
isInstanceOf只能判断出对象是否是指定类及其子类的对象,而不能精确判断出,对象就是指定类的对象。
如果要求精确的判断对象就是指定类的对象,那么就只能使用getClass和classOf了。
对象.getClass可以精确获取对象的类,classOf[类]可以精确获取类,使用==操作即可判断。
6、使用模式匹配进行类型判断
在实际开发中,大量的使用了模式匹配的方式进行类型的判断,这种方式更加简洁明了,而且代码可维护性和可扩展性也非常高。
使用模式匹配,功能性上来说,与isInstanceOf一样,也是判断主要是该类以及该类的子类的对象即可,不是精准判断的。
7、protected
Scala中,可以使用protected修饰field和method,这样在子类中就可以不需要super关键字,直接就可以访问field和method。
可以使用protected[this],则只能在当前子类对象中访问父类的field和method,无法通过其他子类对象访问父类的field和method。
8、调用父类的construct
Scala中,每个类可以有一个主constructor和任意多个辅助constructor,而每个辅助constructor的第一行都必须是调用其它辅助constructor或者是主constructor,因此子类的辅助constructor是一定不能直接调用父类的constructor的。
只能在子类的主constructor中调用父类的constructor,以下这种语法,就是通过子类的主构造函数来调用父类的构造函数。
如果是父类中接受的参数,比如name和age,子类中接受时,就不要用任何val或var来修饰了,否则会认为是子类要覆盖父类的field。
9、匿名内部类
Scala中,匿名子类非常常见,而且非常强大,spark源码中使用了大量这种匿名内部类。
匿名子类就是说可以定义一个类的没有名称的子类,并直接创建其对象,然后将对象的引用赋予一个变量,之后甚至可以将该匿名子类的对象传递给其他函数。
10、抽象类
如果在父类中,某些方法无法立即实现,而需要依赖不同的子类来覆盖,重写实现自己不同的方法实现。此时可以将父类中的这些方法不给出具体的实现,只有方法签名,这种方法就是抽象方法。
如果一个类中有一个抽象方法,那么类就必须用abstract来声明为抽象类,此时抽象类是不可以实例化的。
在子类中覆盖抽象类的抽象方法时,不需要使用override关键字。
11、抽象field
如果在父类中,定义了fiield,但是没有给出初始值,则此field为抽象field。
抽象field意味着,scala会根据自己的规则,为var或val类型的field生成对应的getter和setter方法,但是父类中是没有该field的。
子类必须覆盖field,以定义自己的具体field,并且覆盖抽象field,不需要使用override关键字。