我正在尝试将类构造函数(从备选集)与从解析某些DSL中检索的值列表相匹配。由于这些值是异构的,所以我将它们存储在Array[Any]
中。
我使用下面的一段代码来做到这一点:
val myClassSymbol: ru.ClassSymbol = mirror.classSymbol(Class.forName(myClassName))
val cm: ru.ClassMirror = mirror.reflectClass(myClassSymbol)
val ctor = myClassSymbol.primaryConstructor.alternatives find { c =>
val signature: ru.Type = c.typeSignature
val constructorParams = signature.paramLists.flatten
val constructorParamValues: Seq[Any] = resultOfMyParsing
(constructorParamValues.size == constructorParams.size) && ((constructorParams zip constructorParamValues) forall ((pair: (ru.Symbol, Any)) => {
val sym = pair._1
var param = pair._2
??? // something to match the symbol with the value
}))
}
ctor map {c =>
val ctorm = cm.reflectConstructor(ctor.get.asMethod)
ctorm(resultOfMyParsing: _*)
} getOrElse {
throw new IllegalStateException(s"cannot find ctor for $constructorParamValues") // might be relace with some clever logic as a fallback
}
有谁知道用什么来代替???(或者想出一个更好/更简单的解决方案)
提前感谢!
我认为你通常能做的最好的事情是比较类(由于类型擦除):
val clazz = mirror.runtimeClass(sym.asTerm.typeSignature)
clazz.isInstance(param)
如果你想处理按名称的构造函数参数,这需要稍微修改一下。