0

鉴于以下 JSON...

scala>  val js = Json.parse("""{"key1": "value1", "key2": "value2","list":[{"item1": "value1"},{"item2": "value2"}]}""")
js: play.api.libs.json.JsValue = {"key1":"value1","key2":"value2","list":[{"item1":"value1"},{"item2":"value2"}]}

...我得到这样的第一个元素list

scala> val l = (js \ "list").as[List[JsValue]]
l: List[play.api.libs.json.JsValue] = List({"item1":"value1"}, {"item2":"value2"})

scala> val first = l(0)
first: play.api.libs.json.JsValue = {"item1":"value1"}

...但是如何从list给定索引处删除元素?

4

3 回答 3

1

对此没有任何内容JsValueJsPath您可以使用诸如Monocle之类的 Lens 库。否则,这是一种方法:

(js \ "list").get match {
  case JsArray(items) => dropAt(items, 1)
}

哪里dropAt是:

def dropAt[A](items: Seq[A], id: Int): Seq[A] =
  items.zipWithIndex.filter(_._2 != id).map(_._1)

dropAt不漂亮,但我不知道有什么好的 API。)

于 2015-06-29T19:20:54.817 回答
1

标准集合库中没有dropAt。您可以使用enrich-my-library 模式添加一个。在丰富集合时,您通常最好使用CanBuildFrom结构,这将允许您保留强类型。您可以实现dropAt为:

implicit class TraversableDropAt[A, Repr <: Traversable[A]](val xs: TraversableLike[A, Repr]) extends AnyVal {
  def dropAt[That](n: Int)(implicit cbf: CanBuildFrom[Repr, A, That]): That = {
    val bf = cbf()
    bf.sizeHint(xs.size - 1) 
    bf ++= xs.take(n)
    bf ++= xs.drop(n + 1)
    bf.result
  }
}

这将允许您呼叫myCollection.dropAt(n)任何Traversable分机(例如ListSeqIterable等)。

当您使用 PlayJSON 类型时,通常最好尽快将它们转换为普通的 Scala 类型。Seq[JsValue]在这里,您可以通过多种方式将数组转换为 a :

val items = (js \ "list").as[Seq[JsValue]]
val items = (js \ "list").as[JsArray].value
val items = (js \ "list") match { case JsArray(items) => items }
val JsArray(items) = (js \ "list")

获得items集合后,您可以在其上使用新dropAt方法。

另一种选择是使用上面显示的丰富我的库模式将dropAt方法直接添加到JsArray(或什至)。JsValue

如果需要转回 a JsArray,可以使用该Json.arr方法。

于 2015-06-29T21:46:21.827 回答
0

您可以通过将元素转换为可变集合而不是不可变列表来删除元素。在可变集合上,您可以调用remove(int: Index)将修改集合的操作。所以你的情况可以写成

val js = Json.parse("""{"key1": "value1", "key2": "value2","list":[{"item1": "value1"},{"item2": "value2"}]}""")
val l = (js \ "list").as[ArrayBuffer[JsValue]]

l.remove(0) // returns the removed element at index 0 but modifies the underlying collection l


于 2019-04-24T09:59:43.910 回答