2014-10-13 173 views
0

在运行时获取指定变量的名称让一类在斯卡拉

case class C(val i: Int) { 
    var assignee: Option[String] = None 
} 

和给定的分配给一个名为x变量,

val x = C(1) 
x: C = C(1) 

如何获得价值"x": Stringx.assignee,即

assert(x.assignee == Some("x")) 
+0

如果这个对象是在数组或某些集合? –

+0

@DenisTulskiy好点,只有'val'分配,否则'受让人'仍然是'None'。 – elm

+0

不知道你在问什么。 – tuxdna

回答

1

本地变量名在运行时不可用。但您可以在编译时使用a macro来捕获它。见例如definingValNamehttps://github.com/sbt/sbt/blob/9c442d3aed53bdc89db1ada9d5b204bf02adb339/main/settings/src/main/scala/sbt/std/KeyMacro.scala和它的用法:

def definingValName(c: Context, invalidEnclosingTree: String => String): String = { 
    import c.universe.{ Apply => ApplyTree, _ } 
    val methodName = c.macroApplication.symbol.name 
    def processName(n: Name): String = n.decoded.trim // trim is not strictly correct, but macros don't expose the API necessary 
    def enclosingVal(trees: List[c.Tree]): String = { 
    trees match { 
     case vd @ ValDef(_, name, _, _) :: ts => processName(name) 
     case (_: ApplyTree | _: Select | _: TypeApply) :: xs => enclosingVal(xs) 
     // lazy val x: X = <methodName> has this form for some reason (only when the explicit type is present, though) 
     case Block(_, _) :: DefDef(mods, name, _, _, _, _) :: xs if mods.hasFlag(Flag.LAZY) => processName(name) 
     case _ => 
     c.error(c.enclosingPosition, invalidEnclosingTree(methodName.decoded)) 
     "<error>" 
    } 
    } 
    enclosingVal(enclosingTrees(c).toList) 
} 

def enclosingTrees(c: Context): Seq[c.Tree] = 
    c.asInstanceOf[reflect.macros.runtime.Context].callsiteTyper.context.enclosingContextChain.map(_.tree.asInstanceOf[c.Tree]) 
+0

非常感谢,如何编译它(无法进入REPL而没有错误),以及如何运行它。 – elm