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
中删除一个元素?
JsValue
和 JsPath
中没有任何内容,您可以使用 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
分机(例如 List
、Seq
、Iterable
等)呼叫 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
鉴于以下 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
中删除一个元素?
JsValue
和 JsPath
中没有任何内容,您可以使用 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
分机(例如 List
、Seq
、Iterable
等)呼叫 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