基础知识

1 将trait作为接口使用

此时Trait就与Java中的接口非常类似,不过注意,在Scala中无论继承还是trait,统一都是extends关键字。

Scala跟Java 8前一样不支持对类进行多继承,但是支持多重继承trait,使用with关键字即可

  1. trait HelloTrait{
  2. def sayHello(name: String)
  3. }
  4. trait MakeFriends{
  5. def makeFriends(p: Person)
  6. }
  7. class Person(val name: String) extends HelloTrait with MakeFriends {
  8. def sayHello(name: String) = println("Hello, " + name)
  9. def makeFriends(p: Person) = println("hello " + p.name + ", I'm " + name)
  10. }
  11. defined trait HelloTrait
  12. defined trait MakeFriends
  13. defined class Person
  14. scala> val p = new Person("spark")
  15. p: Person = Person@2f29e630
  16. scala> val p2 = new Person("jack")
  17. p2: Person = Person@52f118aa
  18. scala> p.sayHello("jack")
  19. Hello, jack
  20. scala> p.makeFriends(p2)
  21. hello jack, I'm spark

2 在trait中定义具体方法

Trait不仅可以定以抽象方法,还可以定以具体方法,此时Trait更像是包含了通过工具方法的东西。

有一个专有名词来形容这种情况,叫做Trait功能混入了类

举例:trait中可以包含一些很多类都通用的方法,比如说打印日志等,Spark中就是用了trait来定义了通用的日志打印方法。

  1. trait Logger {
  2. def log(msg: String) = println("log: " + msg)
  3. }
  4. class Person(val name: String) extends Logger {
  5. def sayHello { println("Hello, I'm " + name); log("sayHello is invoked") }
  6. }
  7. defined trait Logger
  8. defined class Person
  9. scala> val p = new Person("leo")
  10. p: Person = Person@3b0c38f2
  11. scala> p.sayHello
  12. Hello, I'm leo
  13. log: sayHello is invoked

3 在trait中定义具体字段

Trait可以定以具体field, 但是这种继承trait field的方式与继承class是原理不同的:如果是继承class获取的field,实际是定以在父类中的;而继承trait获取的field,就直接被添加到了继承类中。 

4 在trait中定义抽象字段

Trait中可以定以抽象field, 而trait中的具体方法可以使用抽象field,但是继承trait的类必须要覆盖抽象field,提供具体的值,否则程序会运行出错。

  1. // trait中的具体方法可以使用抽象field
  2. trait SayHello{
  3. val msg: String
  4. def sayHello(name: String) = println(msg + "," + name)
  5. }
  6. // 继承trait中必须覆盖抽象field
  7. class Person(val name: String) extends SayHello{
  8. val msg: String = "hello"
  9. def makeFriends(p: Person){
  10. sayHello(p.name)
  11. println("I'm" + name + ", want to make friends with you")
  12. }
  13. }
  14. defined trait SayHello
  15. defined class Person
  16. // 测试
  17. scala> val p1 = new Person("leo")
  18. p1: Person = Person@67372d20
  19. scala> val p2 = new Person("Sparks")
  20. p2: Person = Person@4f1f2f84
  21. scala> p1.makeFriends(p2)
  22. hello,Sparks
  23. I'mleo, want to make friends with you

Trait进阶

为实例混入trait

有时我们可以在创建类的对象时,指定该对象混入某个trait,这样就只有这个对象混入该trait的方法,而类的其他对象则没有

  1. trait Logged {
  2. def log(msg: String) {}
  3. }
  4. trait MyLogger extends Logged {
  5. override def log(msg: String) {println("log: " + msg)}
  6. }
  7. class Person (val name: String) extends Logged {
  8. def sayHello { println("Hi, I'm "+ name); log("sayHello is invokend!")}
  9. }
  10. defined trait Logged
  11. defined trait MyLogger
  12. defined class Person
  13. scala> val p1 = new Person("leo")
  14. p1: Person = Person@36f80ceb
  15. scala> p1.sayHello
  16. Hi, I'm leo
  17. // 混入trait,覆盖log方法!
  18. scala> val p2 = new Person("jack") with MyLogger
  19. p2: Person with MyLogger = $anon$1@30a6984c
  20. scala> p2.sayHello
  21. Hi, I'm jack
  22. log: sayHello is invokend!

trait调用链

Scala中支持让类继承多个trait后,依次调用多个trait中的同一个方法(Java中做不到),只要让多个trait的同一个方法中,在最后都执行super.method即可

类中调用多个trait中都有的这个方法时,首先会从最右边的trait方法开始执行,然后依次往左,最终形成一个调用链

这种特性非常强大,其实就相当于设计模式中责任链模式的一种具体实现。

  1. trait Handler{
  2. def handle(data: String) {}
  3. }
  4. trait DataValidHandler extends Handler {
  5. override def handle(data: String){
  6. println("check data:" + data)
  7. // 最后都执行super.method
  8. super.handle(data)
  9. }
  10. }
  11. trait SignatureValidHandler extends Handler {
  12. override def handle(data: String){
  13. println("check signature: " + data)
  14. // 最后都执行super.method
  15. super.handle(data)
  16. }
  17. }
  18. class Person(val name: String) extends SignatureValidHandler with DataValidHandler {
  19. def sayHello = {println("hello, " + name); handle(name)}
  20. }
  21. defined trait Handler
  22. defined trait DataValidHandler
  23. defined trait SignatureValidHandler
  24. defined class Person
  25. scala> val p = new Person("Sparks")
  26. p: Person = Person@4b37d1a4
  27. // 从右往左执行方法
  28. scala> p.sayHello
  29. hello, Sparks
  30. check data:Sparks
  31. check signature: Sparks

混合使用trait的具体方法和抽象方法

可以让具体方法依赖于抽象方法,而抽象方法则放到继承trati的类中去实现

这种trait其实就是设计模式中模板设计模式的体现

  1. trait Valid{
  2. // 将getName交给继承类实现,这里直接在具体方法中使用抽象方法
  3. def getName: String
  4. def valid: Boolean = {
  5. getName == "Sparks"
  6. }
  7. }
  8. class Person(val name: String) extends Valid {
  9. println(valid)
  10. def getName = name
  11. }
  12. defined trait Valid
  13. defined class Person
  14. // 测试
  15. scala> val p = new Person("Sparks")
  16. true
  17. p: Person = Person@351fadfa

trait的构造机制

在Scala中,trait也是有构造代码的,也就是trait中除了method中的所有代码 
而继承了trait的类的构造顺序如下:

  1. 父类的构造函数
  2. trait的构造代码,多个trait从左到右依次执行
  3. 构造trait时先构造父trait,如果多个trait继承同一个父trait,则父trait只会构造一次
  4. 所有trait构造完毕后,自身构造函数执行
  1. class Person{ println("Person's constructor!")}
  2. trait Logger { println("Logger's constuctor!")}
  3. trait MyLogger extends Logger { println("MyLogger's constructor!")}
  4. trait TimeLogger extends Logger { println("TimeLogger constructor")}
  5. class Student extends Person with MyLogger with TimeLogger {
  6. println("Student's constructor")
  7. }
  8. defined class Person
  9. defined trait Logger
  10. defined trait MyLogger
  11. defined trait TimeLogger
  12. defined class Student
  13. // 测试构造顺序
  14. scala> val s = new Student
  15. Person's constructor!
  16. Logger's constuctor!
  17. MyLogger's constructor!
  18. TimeLogger constructor
  19. Student's constructor
  20. s: Student = Student@467421cc

trait field初始化

在Scala中,trait是没有接收参数的构造函数的,这是trait与class的唯一区别,但是如果需求就是要trait能够对field进行初始化,那该怎么办呢?

这时候就需要使用Scala中非常特殊的一种高级特性——提前定义

出错示例:

  1. trait SayHello {
  2. val msg: String
  3. println(msg.toString)
  4. }
  5. class Person extends SayHello{
  6. val msg: String = "init"
  7. }
  8. defined trait SayHello
  9. defined class Person
  10. // 因为要首先初始化trait,但是println中使用了抽象field,所以报错
  11. scala> val p = new Person
  12. java.lang.NullPointerException

  
使用提前定义特性初始化trait field

  1. trait SayHello {
  2. val msg: String
  3. println(msg.toString)
  4. }
  5. class Person
  6. defined trait SayHello
  7. defined class Person
  8. // 提前定义
  9. scala> val p = new {
  10. | val msg: String = "init"
  11. | }with Person with SayHello
  12. init
  13. p: Person with SayHello = $anon$1@445c693
  14. // 提前定义另一种写法
  15. scala> class Person extends {
  16. | val msg: String = "init"
  17. | } with SayHello{}
  18. defined class Person
  19. scala> val p = new Person
  20. init
  21. p: Person = Person@121c1a08

  
使用lazy + override初始化trait field

  1. scala> trait SayHello {
  2. | lazy val msg: String = null
  3. | println(msg.toString)
  4. | }
  5. defined trait SayHello
  6. // 覆盖lazy值
  7. scala> class Person extends SayHello {
  8. | override lazy val msg: String = "init"
  9. | }
  10. defined class Person
  11. scala> val p = new Person
  12. init
  13. p: Person = Person@753c7411

trait继承class

在Scala中,trait也可以继承自class,此时这个class就会成为所有继承该trait的类的父类。

  1. class MyUtil {
  2. def printMessage(msg: String) = println(msg)
  3. }
  4. trait Logger extends MyUtil{
  5. def log(msg: String) = printMessage("log: " + msg)
  6. }
  7. class Person(val name:String) extends Logger{
  8. def sayHello{
  9. log("Hi, I'm" + name)
  10. printMessage("hi,I'm " + name)
  11. }
  12. }
  13. defined class MyUtil
  14. defined trait Logger
  15. defined class Person
  16. scala> val p = new Person("sparks")
  17. p: Person = Person@5bc44d78
  18. // 既可以调用Logger中的方法也可以调用MyUtil中的方法
  19. scala> p.sayHello
  20. log: Hi, I'msparks
  21. hi,I'm sparks

Scala入门系列(八):面向对象之trait的更多相关文章

  1. 微软云平台windows azure入门系列八课程

    微软云平台windows azure入门系列八课程: Windows Azure入门教学系列 (一): 创建第一个WebRole程序与部署 Windows Azure入门教学系列 (二): 创建第一个 ...

  2. C语言高速入门系列(八)

    C语言高速入门系列(八) C语言位运算与文件 本章引言: 在不知不觉中我们的C高速入门系列已经慢慢地接近尾声了,而在这一节中,我们会对 C语言中的位运算和文件进行解析,相信这两章对于一些人来说是陌生的 ...

  3. Scala入门系列(九):函数式编程

    引言 Scala是一门既面向对象,又面向过程的语言,Scala的函数式编程,就是Scala面向过程最好的佐证.也真是因此让Scala具备了Java所不具备的更强大的功能和特性. 而之所以Scala一直 ...

  4. Scala入门系列(六):面向对象之object

    object object相当于class的单个实例,类似于Java中的static,通常在里面放一些静态的field和method.   第一次调用object中的方法时,会执行object的con ...

  5. Scala入门系列(五):面向对象之类

    定义类 // 定义类,包含field以及method class HelloWorld { private var name = "Leo" def sayHello() { pr ...

  6. Scala入门系列(七):面向对象之继承

    extends 与Java一样,也是使用extends关键字,使用继承可以有效复用代码 class Person { private var name = "leo" def ge ...

  7. Scala入门系列(十):函数式编程之集合操作

    1. Scala的集合体系结构 Scala中的集合体系主要包括(结构跟Java相似): Iterable(所有集合trait的根trait) Seq(Range.ArrayBuffer.List等) ...

  8. Scala入门系列(一):基础语法

    Scala基础语法 Scala与JAVA的关系 Scala是基于Java虚拟机,也就是JVM的一门编程语言,所有Scala的代码都需要经过编译为字节码,然后交由Java虚拟机来运行. 所以Scala和 ...

  9. Scala入门系列(四):Map & Tuple

    Map 创建Map // 创建一个不可变的Map scala> val ages = Map("Leo" -> 30, "Sparks" -> ...

随机推荐

  1. php 不写闭合标签

    参阅了一些文章,对PHP闭合标签的总结如下:       好处:如果这个是一个被别人包含的程序,没有这个结束符,可以减少很多很多问题,比如说:header, setcookie, session_st ...

  2. 爆炸快求1~n有多少素数

    这个求一千亿以内的素数大约用6780ms #include <stdio.h> #include <iostream> #include <string.h> #i ...

  3. IdentityServer4 通过 AccessToken 获取 UserClaims

    实现效果:通过生成的access_token获取用户的一些信息,这样客户端请求的时候,不需要传递用户信息了. 示例配置: public void ConfigureServices(IServiceC ...

  4. VS2013装扩展RazorGenerator

    问题:vs2013工具扩展和更新搜索关键字:Razor Generator查找不到,如图 解决:下载网址:https://github.com/RazorGenerator/RazorGenerato ...

  5. phonegap与H5中的接口对比

    接口 HTML5 phonegap 差异 地理定位 geolocation 单次定位: navigator.geolocation.getCurrentPosition(Success, [error ...

  6. SSM框架搭建(Spring+SpringMVC+MyBatis)与easyui集成并实现增删改查实现

    一.用myEclipse初始化Web项目 新建一个web project: 二.创建包 controller        //控制类 service //服务接口 service.impl //服务 ...

  7. C语言 第三章 关系、逻辑运算与分支流程控制

    目录 一.关系运算 二.逻辑运算 三.运算优先级 四.if语句 4.0.代码块 4.1.单if语句 4.2.if else 4.3.多重if 4.4.?号:号表达式 五.switch语句 一.关系运算 ...

  8. 32.Linux-2440下的DMA驱动(详解)

    DMA(Direct Memory Access) 即直接存储器访问, DMA 传输方式无需 CPU 直接控制传输,通过硬件为 RAM .I/O 设备开辟一条直接传送数据的通路,能使 CPU 的效率大 ...

  9. 【爬虫】利用Scrapy抓取京东商品、豆瓣电影、技术问题

    1.scrapy基本了解 Scrapy是一个为了爬取网站数据,提取结构性数据而编写的应用框架.可以应用在包括数据挖掘, 信息处理或存储历史数据等一系列的程序中.其最初是为了页面抓取(更确切来说,网络抓 ...

  10. mybatis批量修改

    使用mybats经常要用到批量修改或者删除,贴出批量修改的代码.如果是批量删除,可将update换成delete. <update id="changestatus" par ...