7

考虑以下json:

{
  "type":"A1",
  "system":{
    "path":"/example.org/FooBar",
    "lastModified":"2013-10-01T12:00:00Z"
  },
  "fields":{
    "foo1":["bar1"],
    "foo2":["bar2"],
    "foo3":["bar3"]
  }
}

现在,使用lift-json,我想把这个json改成:

{
  "type":"A1",
  "system":{
    "path":"/example.org/FooBar",
    "lastModified":"2013-10-01T12:00:00Z"
  },
  "fields":{
    "foo1":["bar1"],
    "foo2":["bar2"],
    "foo3":["bar3"]
  },
  "injected":{
    "bar1":"foo1",
    "bar2":"foo2"
  }
}

所以,我尝试了以下方法:

scala> val json = parse("""
     |{
     |  "type":"A1",
     |  "system":{
     |    "path":"/example.org/FooBar",
     |    "lastModified":"2013-10-01T12:00:00Z"
     |  },
     |  "fields":{
     |    "foo1":["bar1"],
     |    "foo2":["bar2"],
     |    "foo3":["bar3"]
     |  }
     |}""")

json: net.liftweb.json.JValue = JObject(List(JField(type,JString(A1)), JField(system,JObject(List(JField(path,JString(/example.org/FooBar)), JField(lastModified,JString(2013-10-01T12:00:00Z))))), JField(fields,JObject(List(JField(foo1,JArray(List(JString(bar1)))), JField(foo2,JArray(List(JString(bar2)))), JField(foo3,JArray(List(JString(bar3)))))))))

scala> json transform{case JObject(l) => JObject(l ::: List(JField("injected", ("bar1" -> "foo1") ~ ("bar2" -> "foo2"))))}
res0: net.liftweb.json.JsonAST.JValue = JObject(List(JField(type,JString(A1)), JField(system,JObject(List(JField(path,JString(/example.org/FooBar)), JField(lastModified,JString(2013-10-01T12:00:00Z)), JField(injected,JObject(List(JField(bar1,JString(foo1)), JField(bar2,JString(foo2)))))))), JField(fields,JObject(List(JField(foo1,JArray(List(JString(bar1)))), JField(foo2,JArray(List(JString(bar2)))), JField(foo3,JArray(List(JString(bar3)))), JField(injected,JObject(List(JField(bar1,JString(foo1)), JField(bar2,JString(foo2)))))))), JField(injected,JObject(List(JField(bar1,JString(foo1)), JField(bar2,JString(foo2)))))))

scala> Printer.pretty(render(res0))
res1: String = 
{
  "type":"A1",
  "system":{
    "path":"/example.org/FooBar",
    "lastModified":"2013-10-01T12:00:00Z",
    "injected":{
      "bar1":"foo1",
      "bar2":"foo2"
    }
  },
  "fields":{
    "foo1":["bar1"],
    "foo2":["bar2"],
    "foo3":["bar3"],
    "injected":{
      "bar1":"foo1",
      "bar2":"foo2"
    }
  },
  "injected":{
    "bar1":"foo1",
    "bar2":"foo2"
  }
}

如您所见,该injected部分也被添加到fields&system中。我只是想在根目录下添加一次。

那么,我做错了什么?以及如何将 json 转换为我需要的正确结构?

4

2 回答 2

6

您遇到的问题是在 transform 中定义的部分函数在 json 结构中的所有可能级别上都匹配。由于外部 JObject 的“系统”和“字段”组件本身就是 JObject,它们匹配部分函数,​​并且也被转换。

要获得您正在寻找的结果,您需要使匹配更具体,例如:

json transform {case JObject(fields) if (fields contains JField("type", "A1")) =>
   JObject(l ::: ...

使用关于最外层对象的一些唯一信息(这里,它的“类型”设置为“A1”)。

或者,lift json 具有我没有处理过的合并功能,但它可能会提供您想要的:

json merge JObject(JField("injected", ...) :: Nil)
于 2013-10-01T12:15:28.167 回答
4

如果你只需要在一个地方添加一个字段,这transform不是最好的工具——你可以~改用:

val newJson = json match {
  case obj: JObject =>
    obj ~ ("injected" -> ("bar1" -> "foo1") ~ ("bar2" -> "foo2"))
  case _ => throw new RuntimeException("Did not receive a JSON object!")
}

如果你知道你总是会解析一个对象,你可以在这里json转换JObject并避免匹配的业务。

于 2013-10-01T12:20:52.430 回答