类型和泛型
最后更新于:2022-04-01 03:05:47
类型系统的首要目的是检测程序错误。类型系统有效的提供了一个静态检测的有限形式,允许我们代码中明确某种类型的变量并且编译器可以验证。类型系统当然也提供了其他好处,但错误检测是他存在的理由(Raison d’Être)
我们使用类型系统应当反映这一目标,但我们必须考虑到读者(译注:读你代码的人):明智地使用类型可以增加清晰度,而过份聪明只会迷乱。
Scala的强大类型系统是学术探索和实践共同来源(例如[Type level programming in Scala](http://apocalisp.wordpress.com/2010/06/08/type-level-programming-in-scala/)) 。但这是一个迷人的学术话题,这些技术很少在应用和正式产品代码中使用。它们应该被避免。
### 返回类型注解(annotation)
尽管Scala允许返回类型是可以省略的,加上它们提供了很好的文档:这对public方法特别重要。而当一个方法不需要对外暴露,并且它的返回值类型是显而易见的时候,则可以直接省略。
在使用混入(mixin)实例化对象时这一点尤其重要,Scala编译器为这些对象创造了单类。例如:
~~~
trait Service
def make() = new Service {
def getId = 123
}
~~~
上面的make*不需要*定义返回类型为Service;编译器会创建一个加工过的类型: Object with Service{def getId:Int}(译注:with是Scala里的mixin的语法)。若用一个显式的注释:
~~~
def make(): Service = new Service{}
~~~
现在作者则不必改变make方法的公开类型而随意的混入(mix in) 更多的特质(traits),使向后兼容很容易实现。
### 变型
变型(Variance)发生在泛型与子类型化(subtyping)结合的时候。与容器类型的子类型化有关,它们定义了对所包含的类型如何子类型化。因为Scala有声明点变型(declaration site variance)注释(annotation),公共库的作者——特别是集合——必须有丰富的注释器。这些注释对共享代码的可用性很重要,但滥用也会很危险。
不可变(invariants)是Scala类型系统中高级部分,但也是必须的一面,因为它有助于子类型化的应用,应该广泛(并且正确)地使用。
*不可变(Immutable)集合应该是协变的(covariant)*。接受容器化类型得方法应该适当地降级(downgrade)集合:
~~~
trait Collection[+T] {
def add[U >: T](other: U): Collection[U]
}
~~~
*可变(mutable)集合应该是不可变的(invariant)*. 协变对于可变集合是典型无效的。考虑:
~~~
trait HashSet[+T] {
def add[U >: T](item: U)
}
~~~
和下面的类型层级:
~~~
trait Mammal
trait Dog extends Mammal
trait Cat extends Mammal
~~~
如果我现在有一个狗(dog)的 HashSet:
~~~
val dogs: HashSet[Dog]
~~~
把它作为一个哺乳动物的Set,增加一只猫(cat)
~~~
val mammals: HashSet[Mammal] = dogs
mammals.add(new Cat{})
~~~
这将不再是一个只存储狗(dog)的HashSet!
### 类型别名
类型别名应当在其提供了便捷的命名或阐明意图时使用,但对于自解释(不言自明)的类型不要使用类型别名。比如
~~~
() => Int
~~~
比下面定义的别名IntMarker更清晰
~~~
type IntMaker = () => Int
IntMaker
~~~
但,下面的别名:
~~~
class ConcurrentPool[K, V] {
type Queue = ConcurrentLinkedQueue[V]
type Map = ConcurrentHashMap[K, Queue]
...
}
~~~
是有用的,因为它表达了目的并更加简短。
当使用类型别名的时候不要使用子类型化(subtyping)
~~~
trait SocketFactory extends (SocketAddress => Socket)
~~~
SocketFactory 是一个生产Socket的方法。使用一个类型别名更好:
~~~
type SocketFactory = SocketAddress => Socket
~~~
我们现在可以对 SocketFactory类型的值 提供函数字面量(function literals) ,也可以使用函数组合:
~~~
val addrToInet: SocketAddress => Long
val inetToSocket: Long => Socket
val factory: SocketFactory = addrToInet andThen inetToSocket
~~~
类型别名通过用 package object 将名字绑定在顶层:
~~~
package com.twitter
package object net {
type SocketFactory = (SocketAddress) => Socket
}
~~~
注意类型别名不是新类型——他们等价于在语法上用别名代替了原类型。
### 隐式转换
隐式转换是类型系统里一个强大的功能,但应当谨慎地使用。它们有复杂的解决规则, 使得通过简单的词法检查领会实际发生了什么很困难。在下面的场景使用隐式转换是OK的:
* 扩展或增加一个Scala风格的集合
* 适配或扩展一个对象(pimp my library模式)(译注参见:http://www.artima.com/weblogs/viewpost.jsp?thread=179766)
* 通过提供约束证据来加强类型安全。
* 提供了类型的证据 (typeclassing,haskell中的概念,指定义一组函数,其实现因所给的数据类型不同而不同)
* 用于Manifests (注:Manifest[T]包含类型T的运行时信息)
如果你发现自己在用隐式转换,总要问问自己是否不使用这种方式也可以达到目的。
不要使用隐式转换对两个相似的数据类型做自动转换(例如,把list转换为stream);显示地做更好,因为不同类型有不同的语意,读者应该意识到这些含义。 译注: 1)一些单词的意义不同,但翻译为中文时可能用的相似的词语,比如mutable, Immutable 这两个翻译为可变和不可变,它们是指数据的可变与不可变。 variance, invariant 也翻译为 可变和不可变,(variance也翻译为“变型”),它们是指类型的可变与不可变。variance指支持协变或逆变的类型,invariant则相反。