这对我来说是一个非常普遍的情况:
trait CoVariant[+T] {
def bar: T
}
def curried[T](x: String)(y: CoVariant[T]) = y // Not really, but for simplicity
val applied = curried("foo") _
val bar = new CoVariant[String] {
def bar = "bar"
}
applied(bar)
所以我有一些共同变体的东西,我需要先做一些常见的东西,然后让一个通用的部分不应用。我真的希望能够像上面的curried函数一样编写它,但这当然会在最后一行产生编译错误:
- type mismatch; found : CoVariant[String] required: CoVariant[Nothing]
我似乎只是想引入一个类,只是为了让某个地方坚持使用类型参数,所以它可以是抽象的:
trait StupidWrapper {
def apply[T](y: CoVariant[T]) : CoVariant[T]
}
而不是咖喱的东西
def notAsNice(x: String) = new StupidWrapper {
def apply[T](y: CoVariant[T]) = y
}
所以现在:
val applied = notAsNice("foo")
applied(bar)
编译。
这感觉很愚蠢,我猜有更好的方法吗?
更新
我想我会更好地具体化。我有这个:
abstract class ParserToSeq {
def apply[T](parser: Parser[T]): Seq[T]
}
def fromTrainingData(trainingLines: Seq[String]) = new ParserToSeq {
def apply[T](p: Parser[T]) = trainingLines.map(parseAll(p, _)).map {
_ match {
case Success(wt, _) => Some(wt)
case _ => None
}
}.flatten
}
然后
val thisData = fromTrainingData(trainingLines)
lazy val wordTags = thisData(wordtagParser) // Parser[WordTag]
lazy val uniGrams = thisData(uniGramParser) // Parser[UniGram]
…
我只是想取消ParserToSeq并执行:
def fromTrainingData[T](trainingLines: Seq[String])(p: Parser[T]) =
trainingLines.map(parseAll(p, _)).map {
_ match {
case Success(wt, _) => Some(wt)
case _ => None
}
}.flatten
请注意,此处的T特定于每个解析器。我理解的问题是,即使类型实际应用于第二个参数列表,T也会在部分应用的方法上解析为具体类型。
似乎我想在中间某处坚持类型param的声明:
def fromTrainingData(trainingLines: Seq[String])[T](p: Parser[T])
当然,Scala无效。
答案 0 :(得分:2)
正如您所提到的,scala Function
对象don't carry type parameters。
但是,您可以将ParseToSeq
替换为more generic ~>
that takes higher-kinded types,
trait ~>[A[_],B[_]] {
def apply[X](a : A[X]) : B[X]
}
然后你的例子变成,
trait Parser[T]
trait WordTag
trait UniGram
val trainingLines: Seq[String] = ???
val wordtagParser: Parser[WordTag] = ???
val uniGramParser: Parser[UniGram] = ???
def fromTrainingData[T](trainingLines: Seq[String]) = new ~>[Parser,Seq] {
def apply[T](p: Parser[T]): Seq[T] = ???
}
val thisData = fromTrainingData(trainingLines)
lazy val wordTags = thisData(wordtagParser) // Seq[WordTag]
lazy val uniGrams = thisData(uniGramParser) // Seq[UniGram]