Как фильтровать массив объектов по значениям свойств элемента с помощью jq?
Мне нравится фильтровать json файлы с помощью jq:
jq . some.json
Учитывая json, содержащий массив объектов:
{
"theList": [
{
"id": 1,
"name": "Horst"
},
{
"id": 2,
"name": "Fritz"
},
{
"id": 3,
"name": "Walter"
},
{
"id": 4,
"name": "Gerhart"
},
{
"id": 5,
"name": "Harmut"
}
]
}
Я хочу отфильтровать этот список, чтобы отображать только элементы с идентификатором, имеющим значения 2 и 4, поэтому ожидаемый результат:
{
"id": 2,
"name": "Fritz"
},
{
"id": 4,
"name": "Gerhart"
}
Как фильтровать json с помощью jq? Я играл с выбором и картой, но не получил ни одного из них для работы, например:
$ jq '.theList[] | select(.id == 2) or select(.id == 4)' array.json
true
Ответы
Ответ 1
Из документов:
jq '.[] | select(.id == "second")'
Вход [{"id": "first", "val": 1}, {"id": "second", "val": 2}]
Вывод {"id": "second", "val": 2}
Я думаю, вы можете сделать что-то вроде этого:
jq '.theList[] | select(.id == 2 or .id == 4)' array.json
Ответ 2
Вы можете использовать select
внутри map
.
.theList | map(select(.id == (2, 4)))
Или более компактный:
[ .theList[] | select(.id == (2, 4)) ]
Хотя написано, что это немного неэффективно, поскольку выражение дублируется для каждого сравниваемого значения. Это будет более эффективным и, возможно, более удобным для чтения:
[ .theList[] | select(any(2, 4; . == .id)) ]
Ответ 3
Использование select(.id == (2, 4))
здесь обычно неэффективно (см. ниже).
Если ваш jq имеет IN/1
, то его можно использовать для достижения более эффективного решения:
.theList[] | select( .id | IN(2,3))
Если ваш jq не имеет IN/1
, вы можете определить его следующим образом:
def IN(s): first(select(s == .)) // false;
Эффективность
Одним из способов увидеть неэффективность является использование debug
. Например, следующее выражение приводит к 10 вызовам debug
, тогда как на самом деле требуется только 9 проверок равенства:
.theList[] | select( (.id == (2,3)) | debug )
["DEBUG:",false]
["DEBUG:",false]
["DEBUG:",true]
{
"id": 2,
"name": "Fritz"
}
["DEBUG:",false]
["DEBUG:",false]
["DEBUG:",true]
{
"id": 3,
"name": "Walter"
}
["DEBUG:",false]
["DEBUG:",false]
["DEBUG:",false]
["DEBUG:",false]
Индекс /1
В принципе, использование index/1
должно быть эффективным, но на момент написания этой статьи (октябрь 2017 года) его реализация, хотя и быстро (написана на C), неэффективна.
Ответ 4
Вот решение, использующее indices:
.theList | [ .[map(.id)|indices(2,4)[]] ]