我想动态地插入一个类型参数,即List[T],其中T只能在运行时找到。

通常,如果已经有类型参数T(不是这种情况),或者在运行时使用TypeTag,则可以创建这样的绑定。

后者听起来像是要走的路,但这将问题转移到“如何在这里获得TypeTag”。

我知道如何使用类型为T(typeTag[T])或类型为T的实例(请参见下面的getTypeTag)来创建TypeTag。

但是,我都没有。我确实有问题的类型是reflect.runtime.universe.Type。是否有可能将其转换为TypeTag,以便我可以动态插入类型参数?

如果List也可以动态设置,则可加分。

scala> import scala.reflect.runtime.{universe => ru}

scala> def getTypeTag[T: ru.TypeTag](obj: T) = ru.typeTag[T]

scala> def insertTag[T](tt: ru.TypeTag[T]) = List[T]_

scala> val fn = (a: String) => "foo"
fn: String => String = <function1>

scala> val pars = getTypeTag(fn).tpe.typeArgs.init(0)
pars: reflect.runtime.universe.Type = String

scala> insertTag(pars)
<console>:21: error: type mismatch;
 found   : reflect.runtime.universe.Type
 required: reflect.runtime.universe.TypeTag[?]
       insertTag(pars)
                 ^

最佳答案

我还没有想出如何将Type转换为TypeTag。如果您的目标是仅针对函数参数获取TypeTag,则可以修改函数getTypeTag以接受函数:

import scala.reflect.runtime.{universe => ru}
def getTypeTag[T: ru.TypeTag](obj: (T) => Any) = ru.typeTag[T]
def insertTag[T](tt: ru.TypeTag[T]) = List[T] _

val fn = (a: String, b: Int) => "foo"
var parTypeTag = getTypeTag(fn.tupled)
insertTag(parTypeTag) // Seq[(String, Int)] => List[(String, Int)] = <function1>

10-05 23:20