scala之trait详解一
程序员文章站
2022-07-14 11:25:45
...
问题一:scala为什么没有多重继承?
Scala和Java一样不允许从多个超类继承。我们知道,C++允许多重继承,但代价也是出人意料的高。主要是多重继承会出现棱形问题,也叫做钻石问题。关于多重继承的问题,可以参考http://cncc.bingj.com/cache.aspx?q=%E9%92%BB%E7%9F%B3%E9%97%AE%E9%A2%98&d=4527442594041004&mkt=zh-CN&setlang=zh-CN&w=ozz7NgJ0rhkK16PkUrkV6aIBd5PuuXmD在 C++中,需要虚拟基类来解决该问题。读者精读过C++的经典书籍《对象模型》,有兴趣的朋友可以去研究下,还是相当复杂的。这些复杂性,让很多人心生畏惧。
问题二:Scala为什么提供特质(trait)而非接口(interface)?
java的设计者对这些复杂性心生畏惧,采取了非常强的限制策略。
1)类只能扩充一个超类
2)可以实现任意数量的接口,但接口只能包含抽象方法,不能包含字段。
我们通常需要调用其它方法来实现某些方法,很显然,在java中做不到。java中经常看到同时提供接口和抽象基类的做法,但这样做治标不治本。如果你要同时扩展这两个抽象基类呢?
Scala提供特质而非接口。特质可以同时拥有抽象方法和具体方法,儿类可以实现多个特质。这样设计干净利落的解决了Java接口的问题。
问题三:特质可以当接口一样使用吗?
答案是肯定的。Scala特质完全可以像Java的接口那样工作。例如:
[b]
这里我们不需要将方法申明为abstract——特质中未实现的方法默认为是抽象的。
子类可以给出实现:
[b]
如果你需要的特质不只一个,可以用with关键字来添加额外的特质:
[b]
和Java一样,Scala只能有一个超类,但可以有任意数量的特质。
问题四:特质(trait)有什么好的地方?
Trait就像拥有部分实现的接口,它提供了介于单一继承和多重继承的中间地带,因此可以在其它类里混入(mix in)它们,混入的概念可以参考如下连接:http://cncc.bingj.com/cache.aspx?q=scala+%E6%B7%B7%E5%85%A5++%E5%86%B0%E6%B7%87%E6%B7%8B%E5%85%B8%E6%95%85&d=4852335367751471&mkt=zh-CN&setlang=zh-CN&w=3fclnwBatJeeli84tZKeDtUd0r6IUVtB。这样可以用一组特性对类进行加强。
先对Friend类建模,然后将其混入任何类:Man,Woman,Dog等。
假定我们已经建模出Human,想让它成为朋友。朋友是能倾听你说话的人,所以给Human增加一listen方法,如下所示:
[b]
可以看出,朋友这方面的特性不是很突出,而且它被并入的Human类中。开发一段时间后,发现Dog也是人类的好朋友。怎样才能让狗成为人类的好朋友呢?java解决这个问题的方法是创建一个借口Friend,让Human,Dog都实现这个接口。因此,我们不得不在这两个类中提供不同的实现。
这个时候,Scala的trait介入了。trait像一个拥有部分实现的接口。var、val会在混入trait类的内部得到实现。这里注意:定义过而未初始化的val,val被认为是抽象的,需要由混入这些特质的类实现。如下代码所示:
一个类被混入trait之后,通过它的实例可以调用到trait的方法,也可以把它的引用当做trait的引用。
[b]
注意,trait看上去很像类,但还是有很大的区别:
1)trait需要混入类去实现那些已申明的而未初始化的变量和值。
2)它们的构造器不能有任何参数
//如要转载,请留下评论并复制本文链接到文章中
Scala和Java一样不允许从多个超类继承。我们知道,C++允许多重继承,但代价也是出人意料的高。主要是多重继承会出现棱形问题,也叫做钻石问题。关于多重继承的问题,可以参考http://cncc.bingj.com/cache.aspx?q=%E9%92%BB%E7%9F%B3%E9%97%AE%E9%A2%98&d=4527442594041004&mkt=zh-CN&setlang=zh-CN&w=ozz7NgJ0rhkK16PkUrkV6aIBd5PuuXmD在 C++中,需要虚拟基类来解决该问题。读者精读过C++的经典书籍《对象模型》,有兴趣的朋友可以去研究下,还是相当复杂的。这些复杂性,让很多人心生畏惧。
问题二:Scala为什么提供特质(trait)而非接口(interface)?
java的设计者对这些复杂性心生畏惧,采取了非常强的限制策略。
1)类只能扩充一个超类
2)可以实现任意数量的接口,但接口只能包含抽象方法,不能包含字段。
我们通常需要调用其它方法来实现某些方法,很显然,在java中做不到。java中经常看到同时提供接口和抽象基类的做法,但这样做治标不治本。如果你要同时扩展这两个抽象基类呢?
Scala提供特质而非接口。特质可以同时拥有抽象方法和具体方法,儿类可以实现多个特质。这样设计干净利落的解决了Java接口的问题。
问题三:特质可以当接口一样使用吗?
答案是肯定的。Scala特质完全可以像Java的接口那样工作。例如:
[b]
trait Logger{ def log(msg: String) }[/b]
这里我们不需要将方法申明为abstract——特质中未实现的方法默认为是抽象的。
子类可以给出实现:
[b]
class ConsoleLogger extends Logger{//[color=red][b]用extends,而不是implements[/b][/color] def log(msg:String){println(msg)}//[b][color=red]不需要写override[/color][/b] }[/b]
如果你需要的特质不只一个,可以用with关键字来添加额外的特质:
[b]
class ConsoleLogger extends Logger with Cloneable with Serializable[/b]
和Java一样,Scala只能有一个超类,但可以有任意数量的特质。
问题四:特质(trait)有什么好的地方?
Trait就像拥有部分实现的接口,它提供了介于单一继承和多重继承的中间地带,因此可以在其它类里混入(mix in)它们,混入的概念可以参考如下连接:http://cncc.bingj.com/cache.aspx?q=scala+%E6%B7%B7%E5%85%A5++%E5%86%B0%E6%B7%87%E6%B7%8B%E5%85%B8%E6%95%85&d=4852335367751471&mkt=zh-CN&setlang=zh-CN&w=3fclnwBatJeeli84tZKeDtUd0r6IUVtB。这样可以用一组特性对类进行加强。
先对Friend类建模,然后将其混入任何类:Man,Woman,Dog等。
假定我们已经建模出Human,想让它成为朋友。朋友是能倾听你说话的人,所以给Human增加一listen方法,如下所示:
[b]
class Human(var name:String){ def listen()=println("Your friend "+name+" is listening") } class Man(override val name:String) extends Human(name) class Woman(override val name:String) extends Human(name)[/b]
可以看出,朋友这方面的特性不是很突出,而且它被并入的Human类中。开发一段时间后,发现Dog也是人类的好朋友。怎样才能让狗成为人类的好朋友呢?java解决这个问题的方法是创建一个借口Friend,让Human,Dog都实现这个接口。因此,我们不得不在这两个类中提供不同的实现。
这个时候,Scala的trait介入了。trait像一个拥有部分实现的接口。var、val会在混入trait类的内部得到实现。这里注意:定义过而未初始化的val,val被认为是抽象的,需要由混入这些特质的类实现。如下代码所示:
trait Friend{ val name:String def listen()=println("Your friend"+" is listening"); } class Human(val name:String) extends Friend class Dog(val name:String) extends Friend class Man(ovrride val name:String) extends Human class Woman(override val name :String) extends Man class Dog(val name:String)extends Animal with Friend{ override def listen=println(name+"'s listening quietly") }
一个类被混入trait之后,通过它的实例可以调用到trait的方法,也可以把它的引用当做trait的引用。
[b]
val john=new Man("john") val sara=new Woman("sara") val comet=new Dog("comet") john.listen //输出:Your friend john is listening sara.listen //输出:Your Friend sara is listening comet.listen //输出:comet's listening quietly val mansBestFriend:Friend =comet //获取引用 mansBestFreind.listen //comet's listenning quitely def helpAsFriend(friend:Friend)=friend listen helpAsFriend(sara) //输出:Your fiend Sara is listening helpAsFriend(comet) //输出:comet's listening quitely[/b]
注意,trait看上去很像类,但还是有很大的区别:
1)trait需要混入类去实现那些已申明的而未初始化的变量和值。
2)它们的构造器不能有任何参数
//如要转载,请留下评论并复制本文链接到文章中
下一篇: scala之trait详解二:选择性混入
推荐阅读
-
解析Android开发优化之:对界面UI的优化详解(一)
-
Android开发笔记之:一分钟学会使用Logcat调试程序的详解
-
mysql视图之确保视图的一致性(with check option)操作详解
-
详解Docker学习笔记之搭建一个JAVA Tomcat运行环境
-
Laravel框架使用技巧之使用url()全局函数返回前一个页面的地址方法详解
-
Vuejs第一篇之入门教程详解(单向绑定、双向绑定、列表渲染、响应函数)
-
linux命令大全之at命令详解(一次性定时计划任务)
-
详解ASP.NET Core 之 Identity 入门(一)
-
Transactional replication(事务复制)详解之如何跳过一个事务
-
JS库之Three.js 简易入门教程(详解之一)