Play Framework 2.4.1:如何从 JsArray 中删除元素

Play Framework 2.4.1: How to remove an element from a JsArray

鉴于以下 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 中删除一个元素?

JsValueJsPath 中没有任何内容,您可以使用 Lens 库,例如 Monocle。否则,这是一种方法:

(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。)

标准集合库中没有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
  }
}

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

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

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 方法。

另一种选择是使用上面显示的 enrich-my-library 模式将 dropAt 方法直接添加到 JsArray(甚至 JsValue)。

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

您可以通过将元素转换为可变集合而不是不可变列表来删除元素。在可变集合上,您可以调用将修改集合的 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