Подтвердить что ты не робот

Как фильтровать массив объектов по значениям свойств элемента с помощью 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
4b9b3361

Ответ 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)[]] ]