Home > Archives > Scala基础之一阶类型 vs 高阶类型

Scala基础之一阶类型 vs 高阶类型

Published on

最早有一阶类型(first-order type)和高阶类型(higher-kinded type)的概念源于Stackoverflow上的一个关于类型的帖子, 很多答者从不同的角度说明了这两个概念。在我看来类型的基本功能在于抽象,从本质上来讲,这两者的区别在于抽象能力的不同。

Scala的基础问题汇总中提到过一个Console命令:kind用于查看类型信息,我们可以以此为基础进行展开。

scala> :kind -v Int
scala.Int's kind is A
*
This is a proper type.

scala> :kind -v List 
scala.collection.immutable.List's kind is F[+A]
* -(+)-> *
This is a type constructor: a 1st-order-kinded type

scala> :kind -v List[_]
scala.collection.immutable.List's kind is F[+A]
* -(+)-> *
This is a type constructor: a 1st-order-kinded type.

scala> :kind -v Ordering[_]
scala.math.Ordering's kind is F[A]
* -> *
This is a type constructor: a 1st-order-kinded type.

scala> class Ref[M[_]]
scala> :kind -v Ref[List]
Ref's kind is X[F[A]]
(* -> *) -> *
This is a type constructor that takes type constructor(s): a higher-kinded type

在展开之前,解释一个概念Proper type(特定类型)或者说是Monomorphic type(单态类型与之相对应的是多态类型Polymorphic type) ,它指的是那些不需要类型参数的类型,比如说原生类型Int,Long,Double,自定义的类class Foo, class Bar。作为类型构造器的一阶类型和高阶类型,它们都需要传入类型参数才能生成一个类型。

(1) 一阶类型

一阶类型就是针对特定类型的抽象,通过给类型构造器提供特定类型来得到一个具体的(concrete)类型

List
class Ref[T] {} // Ref属于一阶类型
Option

(2) 高阶类型

高阶类型就是在对特定类型抽象的类型的基础上再进行抽象,也就是针对一阶类型的再次抽象; 高阶类型就是对类型构造器进行抽象的类型。

class Ref[M[_]] // Ref属于高阶类型
trait Functor[F[_]] // Functor属于高阶类型
trait Apply[F[_]] // Apply属于高阶类型

对于type MyMap = Map[String, List[String]],MyMap并不是高阶类型,因为不需要传入任何的类型参数; 对于type MyMap[A] = Map[A, List[String]], MyMap也不是高阶类型而是一阶类型,因为A不能再接受类型参数,也就是不再具有抽象的能力。

下面通过上下文定界中的一个实例来进一步理解这两个概念,在Scala基础之类型类(type class pattern)中讲到过上下文定界。

for [A: T],编译器将会尝试寻找形如T[A]的隐式类型实例,在这个地方,我们还需要考虑到T能够接受的类型。

scala> def func[C: ClassTag] = implicitly[ClassTag[C]]
func: [C](implicit evidence$1: scala.reflect.ClassTag[C])scala.reflect.ClassTag[C]
scala> def func[CC[_]: ClassTag] = implicitly[ClassTag[CC]]
<console>:8: error: type CC takes type parameters
       def func[CC[_]: ClassTag] = implicitly[ClassTag[CC]]
                                                       ^

不过我们可以使用类型别名(type alias)来解决这个问题:

type HigherOrdering[CC[T]] = ClassTag[CC[_]]
def func[CC[_]: HigherOrdering] = implicitly[HigherOrdering[CC]]
def func1[CC[_]: HigherOrdering](implicit evidence: HigherOrdering[CC]) = evidence

// 类型lambda
def func2[CC[_]: ({type λ[CC[T]] = ClassTag[CC[_]]})#λ] = implicitly[ClassTag[CC[_]]] 

scala> func2[List]
res0: scala.reflect.ClassTag[List[_]] = scala.collection.immutable.List

scala> func2[Option]
res1: scala.reflect.ClassTag[Option[_]] = scala.Option

scala> func2[Int]
<console>:10: error: Int takes no type parameters, expected: one
              func2[Int]
                    ^

注意第二种写法,func1中evidence的类型是HigherOrdering[CC],而不能是HigherOrdering[CC[_]],HigherOrdering类型需要接受一个可以接受类型参数的类型

参考

> 使用高阶类型定义上下文定界

> 类型构造器的推断

声明: 本文采用 BY-NC-SA 授权。转载请注明转自: Allen写字的地方