在 Scala 中所有值都有一种对应的类型

单例类型

  • 形式:value.type,返回类型 value / null

  • 场景1:链式API调用时的类型指定

    class Super {
    def m1(t: Int) = {println(t); this}
    def m2(t: Int) = {println(t); this}
    }
    // 正常打印
    new Super().m1(1).m2(2) class Child extends Super {
    def c1(t: Int) = {println(t); this}
    } // 异常 value c1 is not a member of Super
    new Child().m1(1).c1(2)

    由于 Scala 会将 this 推断为当前类(即 Super),因此无法完成链式调用

    class Super {
    // 指定返回类型为调用方的 this
    def m1(t: Int): this.type = {println(t); this}
    def m2(t: Int): this.type = {println(t); this}
    } class Child extends Super {
    def c1(t: Int) = {println(t); this}
    } // 成功打印
    new Child().m1(1).c1(2)
  • 场景2:方法中使用 object 实例作为参数

    object Foo
    class Child extends Super {
    def c1(obj: Foo.type) = {
    if (obj == Foo) println("foo")
    this
    }
    }

    Note:不可定义为 def c1(obj: Foo),因为 Foo 为单例对象,而不是类型

类型投影

  • 形式:Outer#Inner

  • 场景:内部类使用时避免类型约束

    class Outer {
    private val inners = ArrayBuffer[Inner]() class Inner (val arg1: Int) {
    val l = ArrayBuffer[Inner]()
    } def add(a: Int) = {
    val t = new Inner(a)
    inners += t
    t
    }
    } val a = new Outer
    val b = new Outer val a1 = a.add(1)
    val b1 = b.add(1)
    a1.l += b1 // error: type mismatch;

    只需要在定义内部类时指定类型投影即可解决

    // 表示适用于任何 Outer 类的 Inner 类
    val l = ArrayBuffer[Outer#Inner]()

    如果将上述例子改用 List 来实现,并不会报错,计算结果也会自动进行类型投射

路径

  • 路径中除最后一部分外,都必须是稳定状态的,如包名、objectvalthis/super/super[S]...

  • 不能包含 var 类型

    var t = new Outer()
    //...其他操作
    val i = new t.Inner // 由于 t 可能会变更,编译器无法确定其含义

    a.b.c.T 内部被翻译成类型投射 a.b.c.type#T

类型别名

  • 形式: type SomeAliasName

  • 必须定义在 classobject 内部

  • 好处: 在引用类型时可以更加简洁

    class Book {
    import scala.collection.mutable._
    // 为该类型取一个别名
    type Index = HashMap[String, Int] // 使用时不在需要重复的定义复杂的数据类型
    val map: Index = new Index()
    } new Book().map // scala.collection.mutable.HashMap[String,Int]

结构类型

  • 为抽象方法、字段、类型的定义某种规范

    def appendLines(target: { def append(str: String): Any },
    lines: Iterable[String]) {
    for (l <- lines) {
    // 此次 Scala 使用反射调用该方法
    target.append(l);
    target.append("\n")
    }
    }

    该方法第一个参数 target 即结构类型,表示使用任何包含该 append 方法的实例作为参数传入。

    由于反射的代价较大,不到万不得已不建议使用,如,有通用行为(append),却无法共享 trait

组合类型 / 交集类型

  • 形式: T1 with T2 with T3 ...

  • 当需要提供多个特质时使用,即用于约束类型

    val image = new ArrayBuffer[java.awt.Shape with java.io.Serializable]
    val rect = new Rectangle(5, 10, 20, 30)
    image += rect // 正确,Rectangle 可序列化
    image += new Area(rect) // 错误 Area 不可序列化
  • 组合类型中也可使用结构类型

    Shape with Serializable { def contains(p: Point): Boolean }

中缀类型

  • 其实只是一种语法写法,如 String Map Int 可代替 Map[String, Int]

  • 可参考数学运算中的表达方式

    type x[A, B] = (String, Int)
    // 即可使用 String x Int 来表示 (String, Int)
  • 一般中缀类型操作符都是左关联的,除了前面提到的 : 操作符,这个是右关联的,比如 List 的操作

中缀类型名称可以是任意操作符,除了 *,避免与类型定义冲突

Scala Types 1的更多相关文章

  1. Scala: Types of a higher kind

    One of the more powerful features Scala has is the ability to generically abstract across things tha ...

  2. Scala Types 2

    存在类型 形式: forSome { type ... } 或 forSome { val ... } 主要为了兼容 Java 的通配符 示例 Array[_] // 等价于 Array[T] for ...

  3. Beginning Scala study note(8) Scala Type System

    1. Unified Type System Scala has a unified type system, enclosed by the type Any at the top of the h ...

  4. scala速成记录1

    选择  Learning Scala这本书,两百多页,足够薄. 安装 http://www.scala-lang.org/  下载Binary的版本.bin里边有所有操作系统下运行的可以运行的交互式s ...

  5. geotrellis使用(十九)spray-json框架介绍

    Geotrellis系列文章链接地址http://www.cnblogs.com/shoufengwei/p/5619419.html 目录 前言 spray-json简介 spray-json使用 ...

  6. Akka(33): Http:Marshalling,to Json

    Akka-http是一项系统集成工具.这主要依赖系统之间的数据交换功能.因为程序内数据表达形式与网上传输的数据格式是不相同的,所以需要对程序高级结构化的数据进行转换(marshalling or se ...

  7. 【原创】大叔问题定位分享(11)Spark中对大表子查询加limit为什么会报Broadcast超时错误

    当两个表需要join时,如果一个是大表,一个是小表,正常的map-reduce流程需要shuffle,这会导致大表数据在节点间网络传输,常见的优化方式是将小表读到内存中并广播到大表处理,避免shuff ...

  8. Spark SQL 函数全集

    org.apache.spark.sql.functions是一个Object,提供了约两百多个函数. 大部分函数与Hive的差不多. 除UDF函数,均可在spark-sql中直接使用. 经过impo ...

  9. org.apache.spark.sql.functions汇总

    测试数据: id,name,age,comment,date 1,lyy,28,"aaa bbb",20180102020325 scala> var data = spar ...

随机推荐

  1. mybatis中集成sharing-jdbc采坑

    1. mybatis中集成sharing-jdbc采坑 1.1. 错误信息 Caused by: org.apache.ibatis.binding.BindingException: Invalid ...

  2. Android源码分析(十六)----adb shell 命令进行OTA升级

    一: 进入shell命令界面 adb shell 二:创建目录/cache/recovery mkdir /cache/recovery 如果系统中已有此目录,则会提示已存在. 三: 修改文件夹权限 ...

  3. 前端开发者必备的Nginx知识

    摘要: 最常用的Web服务器 -- Nginx 原文:前端开发者必备的Nginx知识 作者:ConardLi Fundebug经授权转载,版权归原作者所有. Nginx在应用程序中的作用 解决跨域 请 ...

  4. jqmobile FAQ

    1. 用JavaScript弹出对话框 $("#msgBox").popup("open"); <div data-role="popup&qu ...

  5. MySQL修炼之路四

    1. 外键(foreign key) 1. 定义:让当前表字段的值在另一个表的范围内选择 2. 语法 foreign key(参考字段名) references 主表(被参考字段名) on delet ...

  6. Mycat高可用解决方案一(mysql安装)

    Mycat高可用解决方案一(mysql安装) Mycat关键特性 关键特性 支持SQL92标准 支持MySQL.Oracle.DB2.SQL Server.PostgreSQL等DB的常见SQL语法 ...

  7. Golang 需要避免踩的 50 个坑(二)

    前言 Go 是一门简单有趣的编程语言,与其他语言一样,在使用时不免会遇到很多坑,不过它们大多不是 Go 本身的设计缺陷.如果你刚从其他语言转到 Go,那这篇文章里的坑多半会踩到. 如果花时间学习官方 ...

  8. Python从零开始——安装与运行

  9. centos服务器部署flask项目。

    已安装的环境nginx,python3,mysql,uwsgi,virtualenv 1,创建虚拟环境 virtualenv -p python3 myblog 2,进入虚拟环境 source myb ...

  10. Wpf DataGrid动态添加列,行数据(一)

    由于最近有这方面的需求,而且刚接触wpf不久,在网上找了很多方法,都不是使用MVVM模式的,因为DataGrid的列不能绑定 这就难受了,我想了个折中的方法,这个是使用了MVVMLight的消息机制, ...