Scala 学习(5)之「类——基本概念3」

◇◆丶佛笑我妖孽 提交于 2020-02-26 09:22:11

使用模式匹配进行类型判断

  • 在实际开发中,比如 spark 的源码中,大量的地方都是使用了模式匹配的方式来进行类型的判断,这种方式更加地简洁明了,而且代码得可维护性和可扩展性也非常的高
  • 使用模式匹配,功能性上来说,与 isInstanceOf 一样,也是判断主要是该类以及该类的子类的对象即可,不是精准判断的
class Person
class Student extends Person
val p: Person = new Student

p match {		//模式匹配
	case per: Person => println("it's Person's object")
	case _  => println("unknown type")
}

// 运行后将会出现 “it's Person's object” 的字样

protected

  • 跟 Java 一样,Scala 中同样可以使用 protected 关键字来修饰 field 和 method,这样在子类中就不需要 super 关键字,直接就可以访问父类的 field 和 method
  • 还可以使用 protected[this],则只能在当前子类对象中访问父类的 field 和 method,无法通过其他子类对象访问父类的 field 和 method
class Person {
	protected var name: String = "0mifang"
	protected[this] var hobby: String = "game"
} 
class Student extends Person {
	def sayHello = println("Hello, " + name)
	
	//无法访问到 hobby 变量,因为被 protected[this] 所保护
	def makeFriends(s: Student) {
		println("my hobby is " + hobby + ", your hobby is " + s.hobby)
	}
}

调用父类的constructor

  • Scala 中,每个类可以有一个主 constructor 和任意多个辅助 constructor,而每个辅助 constructor 的第一行都必须是调用其他辅助 constructor 或者是主 constructor;因此子类的辅助 constructor 是一定不可能直接调用父类的 constructor 的
  • 只能在子类的主 constructor 中调用父类的 constructor,以下这种语法,就是通过子类的主构造函数来调用父类的构造函数
  • 注意!如果是父类中接收的参数,比如 name 和 age,子类中接收时,就不要用任何 val 或 var 来修饰了,否则会认为是子类要覆盖父类的 field
class Person(val name: String, val age: Int)
//在主 constructor 中调用父类的 constructor
class Student(name: String, age: Int, var score: Double) extends Person(name, age) {
	def this(name: String) {
		this(name, 0, 0)
	}
	def this(age: Int) {
		this("0mifang", age, 0)
	}
}

匿名内部类

匿名子类,也就是说,可以定义一个类的没有名称的子类,并直接创建其对象,然后将对象的引用赋予一个变量。之后甚至可以将该匿名子类的对象传递给其他函数。

class Person(protected val name: String) {
	def sayHello = "Hello, I'm " + name
}
val p = new Person("0mifang") {	//匿名内部类
	override def sayHello = "Hi, I'm " + name
}
def greeting(p: Person { def sayHello: String }) {	//使用匿名内部类作为参数
	println(p.sayHello)
}

抽象类

  • 如果在父类中,有某些方法无法立即实现,而需要依赖不同的子类来覆盖,重写实现自己不同的方法实现。此时可以将父类中的这些方法不给出具体的实现,只有方法签名,这种方法就是抽象方法
  • 而一个类中如果有一个抽象方法,那么类就必须用 abstract 来声明为抽象类,此时抽象类是不可以实例化的
  • 在子类中覆盖抽象类的抽象方法时,不需要使用 override 关键字
abstract class Person(val name: String) {
	def sayHello: Unit		//无具体实现
}
class Student(name: String) extends Person(name) {
	def sayHello: Unit = println("Hello, " + name)		//覆盖并做出具体实现
}

抽象field

  • 在父类中,定义了 field,但是没有给出初始值,则此 field 为抽象 field
  • 抽象 field 意味着,Scala 会根据自己的规则,为 var 或 val 类型的 field 生成对应的 getter 和 setter 方法,但是父类中是没有该 field 的,子类必须覆盖 field,以定义自己的具体 field,并且,覆盖抽象 field,不需要使用 override 关键字
abstract class Person {
	val name: String	//无初始值
}
class Student extends Person {
	val name: String = "0mifang"	//覆盖并给出初始值
}

欢迎关注,本号将持续分享本人在编程路上的各种见闻。

标签
易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!