多个trait的加载顺序
如果有多个父类,则按照从左到右的顺序调用。
例如:
package t2
/*
多继承时,构造器调用顺序
1.先父后子。先调用父类构造器,再调用子类构造器
2.有多个父类,从左到右的顺序去调用构造器
*/
object test20 {
trait A {
println("trait A")
}
trait B {
println("trait B")
}
trait C {
println("trait B")
}
class Class1 extends A with B with C{
println("class 1...")
}
def main(args: Array[String]): Unit = {
new Class1()
}
}
package t2
/*
多继承时,构造器调用顺序
1.先父后子。先调用父类构造器,再调用子类构造器
2.有多个父类,从左到右的顺序去调用构造器
*/
object test20 {
trait A {
println("trait A")
}
trait BB {
println("trait BB")
}
trait B extends BB{
println("trait B")
}
trait CC {
println("trait CC")
}
trait C extends CC{
println("trait C")
}
class Class1 extends A with B with C{
println("class 1...")
}
def main(args: Array[String]): Unit = {
new Class1()
}
}
空指针异常
通过打印,引导学生找到问题:调用p.log()时,fileName没有值。这就是继承时带的问题:先执行了trait构造器的代码,后执行了具体子类的构造器。而具体的赋值操作是在子类的构造器中才进行,所以,父类的filename没有值,导致空指针异常。
package t2
import java.io.FileWriter
/*
空指针异常
目标:实现日志类,可以把一些文本信息写入指定的文件中
*/
object test22 {
trait Log {
//文件名,抽象属性
var fileName:String
println(s"${fileName}")
lazy val fileWriter:FileWriter=new FileWriter(fileName)
def writeLog(msg:String):Unit={
fileWriter.write(msg)
fileWriter.close()
}
}
class TextLog extends Log{
override var fileName:String="text01.log"
}
def main(args: Array[String]): Unit = {
val textLog=new TextLog()
textLog.writeLog("2025-11-19:下午,天气晴朗")
}
}