如何在Scala中编写多态输入函数?

时间:2012-06-14 19:06:02

标签: arrays scala input polymorphism

我正在尝试在Scala中编写一些便利函数来读取值数组。

我开始使用一个将“1 1 2 3 5 8”之类的字符串转换为数组[Int]的函数:

def readInts(in: String) = in.split(" ").map(_.toInt)

这很好用,除非我想读不仅仅是Ints而是Longs或BigInts或Doubles,我需要为每一个定义一个函数,这看起来很浪费(特别是如果我推广到读取矩阵或其他复合数据) )

我希望能够编写单个多态函数,如下所示:

def readArray[A](in: String) = in.split(" ").map(_.to[A])

据我所知,这是不可能的,因为String类没有多态'to'方法。好的;我将尝试将其定义为辅助方法:

def to[A](in: String) = ???

似乎我需要在类型参数上有条件地定义方法 - 如果A是Int,则调用in.toInt;如果A为Double,请致电in.toDouble;如果A是Tuple2 [Int,Int],则调用辅助方法toTupleOfInts(in)。据我所知,这也是不可能的。

在我知道的另一种函数语言Haskell中,这个问题由'Read'类型类处理,它定义了一个多态函数'read',它从String转换为所需的数据类型。

在Scala中执行此操作(即写入多态输入函数)的惯用方法是什么?

1 个答案:

答案 0 :(得分:9)

你可以做一些非常接近Haskell类型类的东西。但是,它不能自动导出(至少在某些未来的版本中可能允许使用宏)

首先,定义一个特征,相当于类型类。

trait Read[A] {
  def read(in: String): A
}

然后隐式提供一些实例,最好是在伴随对象

object Read {

  implicit object ReadInt extends Read[Int] {
    def read(in: String): Int = in.toInt
  }

  implicit object ReadDouble ....

  implicit def readArray[A](implicit readItem: Read[A]) : Read[Array[A]]
    = new Read[Array[A]] {
      def read(in: String) = in.split(" ").map(readItem.read _)
    }

   implicit def readTuple[A,B](implicit readA: Read[A], readB: Read[B]) ...

}

最后,定义一个使Read易于访问的方法

def read[A](in: String[A])(implicit reader: Read[A]) = reader.read(in)

您可以在隐式作用域中存在Read实例的任何类型上调用read。