作为 Circe-Json 的一部分,如何使用 Hcursor 或 Optics 到 return 匹配对象列表?

How to use Hcursor or Optics, as part of Circe-Json, to return a List of matching Objects?

我的代码大致如下所示:

val json: Json = parse("""
[
{
"id": 1,
"type": "Contacts",
"admin": false,
"cookies": 3
},
{
"id": 2,
"type": "Apples",
"admin": false,
"cookies": 6
},
{
"id": 3,
"type": "Contacts",
"admin": true,
"cookies": 19
}
]
""").getOrElse(Json.Null)

我正在使用 Circe、Cats、Scala、Circe-json 等,Parse 调用成功。

我想要 return 一个列表,其中 type="Contacts" 的每个顶级对象都完整显示。

类似于: List[String] = ["{"id": 1,"type": "Contacts","admin": false,"cookies": 3}","{"id": 3,"type": "Contacts","admin": 真,"cookies": 19}"]

背景是我的磁盘上有很大的 JSON 文件。我需要过滤掉匹配特定 type= 值的对象子集,在本例中为 type=Contacts,然后将它们从 json 文件的其余部分中分离出来。我不想修改文件,我更希望通过 grep 查找匹配的对象并相应地处理它们。

谢谢。

完成这种事情最直接的方法是将文档解码为 List[Json]List[JsonObject] 值。例如,给定您对 json:

的定义
import io.circe.JsonObject

val Right(docs) = json.as[List[JsonObject]]

然后可以根据type:

查询
scala> val contacts = docs.filter(_("type").contains(Json.fromString("Contacts")))
contacts: List[io.circe.JsonObject] = List(object[id -> 1,type -> "Contacts",admin -> false,cookies -> 3], object[id -> 3,type -> "Contacts",admin -> true,cookies -> 19])

scala> contacts.map(Json.fromJsonObject).map(_.noSpaces).foreach(println)
{"id":1,"type":"Contacts","admin":false,"cookies":3}
{"id":3,"type":"Contacts","admin":true,"cookies":19}

鉴于您的用例,circe-optics 似乎不太适合(请参阅 讨论为什么使用任意谓词进行过滤与 Monocle 的 Traversal 很尴尬)。

不过,如果您对解析和过滤大型 JSON 文件而不将文件的全部内容加载到内存中感兴趣,那么可能值得研究一下 circe-fs2 或 circe-iteratee。在这两种情况下,原理都与上面的 List[JsonObject] 代码相同 — 您将大 JSON 数组解码为 JsonObject 值流,您可以根据需要对其进行查询。