简介
- trait,中文意思"特质"。
- scala中叫"特质", 有的人也叫"特征"。
- scala中的trait相当于java中的"接口+抽象类"
- java的抽象类既可以有抽象方法也可以有非抽象方法,而接口中只能有抽象方法,接口属于抽象类的一个特例(即接口就是只有抽象方法的抽象类)!
- 那么问题就来了,既然接口是抽象类的一个特例,那么为什么非要高出两个概念(抽象类和接口),而且关键字也要用两个:abstract 和 interface, 这样岂不是增加了java的学习难度?
- 原因就在于java是单继承的,只能继承一个类,所以搞出了一个"接口"的概念,允许实现(implement)多个接口!
- 那么scala就很好滴解决了这个问题:, 提出一个全新的trait(特质)概念, 这一个概念就相当于java的"接口+抽象类"。 那么同理,很容易可以想到,这个trait中,是既可以写抽象方法,也可以写非抽象方法!
ps:据说java8的最新的编译器,已经允许在接口中写非抽象方法了,编译器也不会报错的,这个有待查证。
那么就又引出了新问题:我写一个scala的trait,最终肯定要编译成class文件,然后放到jvm中运行, 然后如果你的jvm编译器不是最新的,那么他就不会允许你在接口里写非抽象方法,然后运行就一定会失败的!
这里必须要强调一个关键因素:trait仅仅是scala中的概念!
那为什么scala的代码能在jvm上正常运行呢?
原因就是咱们的java或者scala都是编译型语言,你必须要先编译后运行!
jvm运行的是.class文件,而不是.java文件或者.scala文件!
java 是通过javac命令来编译, scala是通过scalac命令来编译!
所以同理,如果你自己发明一个新语言,假如说就叫A语言,然后你再用Ac命令去编译,只要编译出来的.class文件符合jvm的规范,那么你的A语言就能在jvm上运行! 而A语言的语法和概念都是你自己来定义的!
所以,为什么scala的trait能在jvm上运行? 关键原因在于scala的编译器!
那么我们就写一个scala的trait, 然后用scalac命令编译一下,看看他到底生成了什么样的.class文件,然后再反编译一下.class文件,用jd_gui工具反编译出来是java的语法,然后咱就可以用java来理解scala的trait。
- 综上:如上述所言,trait是scala中的抽象类,那么既然它是类,那么它就可以再继承别的类,即scala的trait可以继承一个class:
trait A extends B{
// trait继承了class, 那么此时trait也有了sayHello函数了。这就是对trait的扩展。
}
class B {
def sayHello():Unit={
println("Hello")
}
}