Size: 2524
Comment: clean up
|
Size: 4255
Comment: DontEverDoThese does not exist as such
|
Deletions are marked like this. | Additions are marked like this. |
Line 1: | Line 1: |
[[Anchor(faq18)]] | <<Anchor(faq18)>> |
Line 7: | Line 7: |
# Bash | |
Line 15: | Line 16: |
00 | |
Line 22: | Line 22: |
This gets tedious for large sequences, but there are other ways, too. If you have the {{{printf}}} command (which is a Bash builtin, and is also POSIX standard), it can be used to format a number: |
The same thing can be accomplished more concisely like this: |
Line 26: | Line 25: |
for ((i=1; i<=10; i++)) # Bash 2 for-loop syntax | # Bash for i in 0{1..9} 10 do echo $i done }}} Another example, for output of 0000 to 0034: {{{ # Bash for i in {000{0..9},00{10..34}} do echo $i done # using the outer brace instead of just adding them one next to the other # allows to use the expansion, for instance, like this: wget 'http://foo.com/adir/thepages'{000{0..9},00{10..34}}'.html' }}} This gets tedious for large sequences, but there are other ways, too. If you have the {{{printf}}} command (which is a Bash builtin, and is also POSIX standard), it can be used to format a number: {{{ # Bash for ((i=1; i<=10; i++)) |
Line 32: | Line 55: |
In Bash 3, you can use ranges inside brace expansion. Also, since {{{printf}}} will implicitly loop if given more arguments than format specifiers, you can simplify this enormously: |
In Bash 3, you can use ranges inside brace expansion. Also, since {{{printf}}} will implicitly loop if given more arguments than format specifiers, you can simplify this enormously: {{{ # Bash 3 printf "%03d\n" {1..300} }}} If you don't know in advance what the starting and ending values are: {{{ # Bash 3 # START and END are variables containing integers eval printf '"%03d\n"' {$START..$END} }}} The `eval` is needed here because you cannot have variables in a brace expansion -- only constants. The extra quotes are required by the `eval` so that our `\n` isn't changed to an `n`. Given how messy that `eval` solution is, please give serious thought to using the `for` loop instead. Also, bash version 4 allows zero-padding in its brace expansion: |
Line 36: | Line 73: |
printf "%03d\n" {1..300} # Bash 3 brace expansion | # Bash 4 echo {01..10} for i in {01..10}; do ... |
Line 39: | Line 79: |
The KornShell and KornShell93 have the {{{typeset}}} command to specify the number of leading zeros: |
The KornShell has the {{{typeset}}} command to specify the number of leading zeros: |
Line 42: | Line 81: |
# Korn | |
Line 48: | Line 88: |
Line 54: | Line 93: |
Line 60: | Line 98: |
Line 62: | Line 99: |
# POSIX shell, GNU utilities | |
Line 67: | Line 105: |
Finally, the following example works with any BourneShell derived shell to zero-pad each line to three bytes: | Be warned however that `seq` might be considered bad style; it's even mentioned in [[BashGuide/Practices/#Don.27t_Ever_Do_These|Don't Ever Do These]]. |
Line 69: | Line 107: |
Finally, the following example works with any BourneShell derived shell (which also has `expr` and `sed`) to zero-pad each line to three bytes: | |
Line 70: | Line 109: |
# Bourne | |
Line 81: | Line 121: |
Now, since the number one reason this question is asked is for downloading images in bulk, you can use the {{{printf}}} command with {{{xargs(1)}}} and {{{wget(1)}}} to fetch files: |
But if you're going to rely on an external Unix command, you might as well just do the whole thing in `awk` in the first place: |
Line 84: | Line 123: |
printf "%03d\n" {$START..$END} | xargs -i% wget $LOCATION/% | # Bourne # COUNT variable contains an integer awk "BEGIN {for (i=1;i<$COUNT;i++) {printf(\"%03d\n\",i)} }" |
Line 87: | Line 128: |
Or, in a slightly more general case: | Now, since the number one reason this question is asked is for downloading images in bulk, you can use the examples above with {{{xargs(1)}}} and {{{wget(1)}}} to fetch files: {{{ almost any example above | xargs -i% wget $LOCATION/% }}} |
Line 89: | Line 133: |
The `xargs -i%` will read a line of input at a time, and replace the `%` at the end of the command with the input. Or, a simpler example using a `for` loop: |
|
Line 90: | Line 137: |
# Bash 3 | |
Line 95: | Line 143: |
---- CategoryShell |
How can I use numbers with leading zeros in a loop, e.g. 01, 02?
As always, there are different ways to solve the problem, each with its own advantages and disadvantages.
If there are not many numbers, BraceExpansion can be used:
# Bash for i in 0{1,2,3,4,5,6,7,8,9} 10 do echo $i done
Output:
01 02 03 [...]
The same thing can be accomplished more concisely like this:
# Bash for i in 0{1..9} 10 do echo $i done
Another example, for output of 0000 to 0034:
# Bash for i in {000{0..9},00{10..34}} do echo $i done # using the outer brace instead of just adding them one next to the other # allows to use the expansion, for instance, like this: wget 'http://foo.com/adir/thepages'{000{0..9},00{10..34}}'.html'
This gets tedious for large sequences, but there are other ways, too. If you have the printf command (which is a Bash builtin, and is also POSIX standard), it can be used to format a number:
# Bash for ((i=1; i<=10; i++)) do printf "%02d " "$i" done
In Bash 3, you can use ranges inside brace expansion. Also, since printf will implicitly loop if given more arguments than format specifiers, you can simplify this enormously:
# Bash 3 printf "%03d\n" {1..300}
If you don't know in advance what the starting and ending values are:
# Bash 3 # START and END are variables containing integers eval printf '"%03d\n"' {$START..$END}
The eval is needed here because you cannot have variables in a brace expansion -- only constants. The extra quotes are required by the eval so that our \n isn't changed to an n. Given how messy that eval solution is, please give serious thought to using the for loop instead.
Also, bash version 4 allows zero-padding in its brace expansion:
# Bash 4 echo {01..10} for i in {01..10}; do ...
The KornShell has the typeset command to specify the number of leading zeros:
# Korn $ typeset -Z3 i=4 $ echo $i 004
If the command seq(1) is available (it's part of GNU sh-utils/coreutils), you can use it as follows:
seq -w 1 10
or, for arbitrary numbers of leading zeros (here: 3):
seq -f "%03g" 1 10
Combining printf with seq(1), you can do things like this:
# POSIX shell, GNU utilities printf "%03d\n" $(seq 300)
(That may be helpful if your version of seq(1) lacks printf-style format specifiers. Since it's a nonstandard external tool, it's good to keep your options open.)
Be warned however that seq might be considered bad style; it's even mentioned in Don't Ever Do These.
Finally, the following example works with any BourneShell derived shell (which also has expr and sed) to zero-pad each line to three bytes:
# Bourne i=0 while test $i -le 10 do echo "00$i" i=`expr $i + 1` done | sed 's/.*\(...\)$/\1/g'
In this example, the number of '.' inside the parentheses in the sed command determines how many total bytes from the echo command (at the end of each line) will be kept and printed.
But if you're going to rely on an external Unix command, you might as well just do the whole thing in awk in the first place:
# Bourne # COUNT variable contains an integer awk "BEGIN {for (i=1;i<$COUNT;i++) {printf(\"%03d\n\",i)} }"
Now, since the number one reason this question is asked is for downloading images in bulk, you can use the examples above with xargs(1) and wget(1) to fetch files:
almost any example above | xargs -i% wget $LOCATION/%
The xargs -i% will read a line of input at a time, and replace the % at the end of the command with the input.
Or, a simpler example using a for loop:
# Bash 3 for i in {1..100}; do wget "$prefix$(printf %03d $i).jpg" # other commands done