[7] - trait

作者: 牛肉圆粉不加葱 | 来源:发表于2016-05-31 23:59 被阅读293次

当你开始使用继承来重用代码时,你入门了;当你开始避免使用继承来重用代码时,你成熟了

这是我以前在知乎上看到关于类继承作用的回答,虽不完全正确,却十分明确的表达出了好的代码应避免类继承而尽量使用类组合。Scala 显然也非常赞同这一点,以至于有了 trait,又叫做特质。当我们定义特质时,应该要遵循这样的原则:一个 trait 只干一件事,如果要干多件事,就定义多个 trait,然后使用一个类来 extends 这些 traits

定义 trait

trait 的定义与 class 类似:

scala> trait T {
     | }
defined trait T

当然,trait 可以包含成员和方法,并且:

  • trait 中的成员可以仅声明,也可以声明并指定值
  • trait 中的方法可以有实现,也可以只有声明而没有实现
scala> trait T {
     |   val a: Int
     |   val b: Int = 1
     |
     |   def getA(): Int
     |   def getB() = b
     | }
defined trait T

对比而言,类一旦包含未定义的方法就必须声明为 abstract;而 Java 的接口中的方法是不能实现的,必须是抽象方法。如果 trait 既为实现它所声明的方法,也没有定义或声明其他成员,那么在字节码级别,该 trait 其实是接口是相同的

另一个与类不同的是,trait 主构造函数不允许有参数列表,并且不允许为 trait 定义辅助构造函数

混入多个 trait

Scala 类只能有一个父类,但可以混入多个 trait,当要混入多个 traits 或已经继承了某个父类时,需要使用关键字 with,如下例:

scala> trait T {
     |   val a: Int
     |   val b: Int = 1
     |
     |   def getA(): Int
     |   def getB() = b
     | }
defined trait T

scala>

scala> trait Q {
     |   def currentTime: String = System.currentTimeMillis().toString
     | }
defined trait Q

scala>

scala> class X extends T with Q {
     |   override val a = 1
     |   override def getA(): Int = a
     | }
defined class X

当类混入 trait 时,需要实现 trait 中为实现的成员和方法。要混入多个 trait 是为了保证『高内聚』,通俗说就是一个 trait 只干一件事,如果要干多件事,就定义多个 trait 然后混入它们


当你继承的父类和混入的特质或混入的不同特质之间有同名方法时可能会有冲突,分为以下几种情况:

  • trait 中的方法未实现:不会冲突
scala> class C {
     |   def a: String = "a"
     | }
defined class C

scala>

scala> trait T {
     |   def a: String
     | }
defined trait T

scala>

scala> trait Q extends C with T {}
defined trait Q
  • trait 中的方法实现了且与父类中的方法参数列表及返回类型相同:会冲突
scala> class C {
     |   def a: String = "a"
     | }
defined class C

scala>

scala> trait T {
     |   def a: String = ""
     | }
defined trait T

scala>

scala> trait Q extends C with T {}
<console>:9: error: trait Q inherits conflicting members:
  method a in class C of type => String  and
  method a in trait T of type => String
(Note: this can be resolved by declaring an override in trait Q.)
       trait Q extends C with T {}
             ^
  • trait 中的方法实现了且与父类中的参数列表相同,返回类型不同:会冲突
scala> class C {
     |   def a: String = "a"
     | }
defined class C

scala>

scala> trait T {
     |   def a: Int = 1
     | }
defined trait T

scala>

scala> trait Q extends C with T {}
<console>:9: error: trait Q inherits conflicting members:
  method a in class C of type => String  and
  method a in trait T of type => Int
(Note: this can be resolved by declaring an override in trait Q.)
       trait Q extends C with T {}
             ^
  • trait 中的方法实现了且与父类的参数列表不同,返回类型相同:不会冲突
scala> class C {
     |   def a: String = "a"
     | }
defined class C

scala>

scala> trait T {
     |   def a( i: Int ): String = i.toString
     | }
defined trait T

scala>

scala> trait Q extends C with T {}
defined trait Q

trait 的继承

一个 trait 同样可以混入其他 trait 或继承类:

scala> class C {
     |   def currentTime: String = System.currentTimeMillis().toString
     | }
defined class C

scala>

scala> trait T {
     |   def random: Int
     | }
defined trait T

scala>

scala> trait Q extends C with T {}
defined trait Q

虽然 Scala 语言支持你这么做,但我个人并不推荐


**传送门: **Scala 在简书目录


欢迎关注我的微信公众号:FunnyBigData

FunnyBigData

相关文章

  • [7] - trait

    当你开始使用继承来重用代码时,你入门了;当你开始避免使用继承来重用代码时,你成熟了 这是我以前在知乎上看到关于类继...

  • rust-trait

    什么是trait,trait相对于java就是interface。 基本的trait例子 在trait中,由sel...

  • 知识点65 :T字母开头的单词 trait-turbulen

    trait 个性特点A trait is a particular characteristic, quality...

  • Scala编程详解12:面向对象编程之Trait

    大纲1、trait基础知识1-1 将trait作为接口使用1-2 在trait中定义具体方法1-3 在trait中...

  • php中的trait机制提高代码复用

    什么是trait 自 PHP 5.4.0 起,PHP 实现了一种代码复用的方法,称为 trait。 Trait 是...

  • Copy Trait 和clone Trait

    1. copy trait语法声明:#[lang = "copy"]pub trait Copy: Clone ...

  • 7.9 Trait

    trait 是PHP为类只能单继承而实现的代码复用机制。 trait 不能继承,也不能实现接口。 trait 不能...

  • Rust范型(2) - 范型约束

    trait中的范型 在定义trait时,可以使用范型: Dagongren是一个trait,包含了banzhuan...

  • Rust impl trait

    trait特性 trait特性可以理解为Java中的接口,具备和接口很类似的特性。trait中的函数叫做方法。某个...

  • Xcode8新功能Trait Variations

    显然Trait Variations替换了以前Xcode7的Size Class(尺寸分类,也叫做自适应布局)。可...

网友评论

    本文标题:[7] - trait

    本文链接:https://www.haomeiwen.com/subject/flhhdttx.html