1
0
mirror of https://github.com/stedolan/jq.git synced 2024-05-11 05:55:39 +00:00

Add until(cond; next); fix #639

This commit is contained in:
Nicolas Williams
2014-12-12 17:35:59 -06:00
parent 2075bec60f
commit be11b2768f
3 changed files with 26 additions and 0 deletions

View File

@@ -1090,6 +1090,10 @@ static const char* const jq_builtins[] = {
" def _while: "
" if cond then ., (update | _while) else empty end; "
" try _while catch if .==\"break\" then empty else . end;",
"def until(cond; next): "
" def _until: "
" if cond then . else (next|_until) end;"
" _until;",
"def limit($n; exp): if $n < 0 then exp else foreach exp as $item ([$n, null]; if .[0] < 1 then break else [.[0] -1, $item] end; .[1]) end;",
"def first(g): foreach g as $item ([false, null]; if .[0]==true then break else [true, $item] end; .[1]);",
"def last(g): reduce g as $item (null; $item);",

View File

@@ -1270,6 +1270,24 @@ sections:
input: '1'
output: ['[1,2,4,8,16,32,64]']
- title: "`until(cond; next)`"
body: |
The `until(cond; next)` function allows you to repeatedly
apply the expression `next`, initially to `.` then to its own
output, until `cond` is true. For example, this can be used
to implement a factorial function (see below).
Note that `until(cond; next)` is internally defined as a
recursive jq function. Recursive calls within `until()` will
not consume additional memory if `next` produces at most one
output for each input. See advanced topics below.
examples:
- program: '[.,1]|until(.[0] < 1; [.[0] - 1, .[1] * .[0]])|.[1]'
input: '4'
output: ['24']
- title: "`recurse(f)`, `recurse`, `recurse(f; condition)`, `recurse_down`"
body: |

View File

@@ -250,6 +250,10 @@ null
1
[1,2,4,8]
[.[]|[.,1]|until(.[0] < 1; [.[0] - 1, .[1] * .[0]])|.[1]]
[1,2,3,4,5]
[1,2,6,24,120]
[foreach .[] as $item ([3, null]; if .[0] < 1 then break else [.[0] -1, $item] end; .[1])]
[11,22,33,44,55,66,77,88,99]
[11,22,33]