bash:files:read_a_file
Differences
This shows you the differences between two versions of the page.
Both sides previous revisionPrevious revisionNext revision | Previous revision | ||
bash:files:read_a_file [2021/01/26 13:56] – [Field splitting, white-space trimming, and other input processing] peter | bash:files:read_a_file [2021/01/26 14:19] (current) – [How to keep other commands from "eating" the input] peter | ||
---|---|---|---|
Line 6: | Line 6: | ||
[[BASH: | [[BASH: | ||
+ | |||
+ | [[BASH: | ||
[[BASH: | [[BASH: | ||
+ | |||
+ | [[BASH: | ||
[[BASH: | [[BASH: | ||
Line 13: | Line 17: | ||
[[BASH: | [[BASH: | ||
+ | [[BASH: | ||
- | ---- | ||
- | |||
- | |||
- | ====== Input source selection ====== | ||
- | |||
- | The redirection < " | ||
- | |||
- | If your input source is the contents of a variable/ | ||
- | |||
- | <code bash> | ||
- | while IFS= read -r line; do | ||
- | printf ' | ||
- | done <<< | ||
- | </ | ||
---- | ---- | ||
- | The same can be done in any Bourne-type shell by using a "here document" | ||
- | |||
- | <code bash> | ||
- | while IFS= read -r line; do | ||
- | printf ' | ||
- | done <<EOF | ||
- | $var | ||
- | EOF | ||
- | </ | ||
- | |||
- | ---- | ||
- | |||
- | ====== Read from a command instead of a regular file ====== | ||
- | |||
- | <code bash> | ||
- | some command | while IFS= read -r line; do | ||
- | printf ' | ||
- | done | ||
- | </ | ||
- | |||
- | ---- | ||
- | |||
- | This method is especially useful for processing the output of find with a block of commands: | ||
- | |||
- | <code bash> | ||
- | find . -type f -print0 | while IFS= read -r -d '' | ||
- | mv " | ||
- | done | ||
- | </ | ||
- | |||
- | **NOTE: | ||
- | |||
- | * **-print0**: | ||
- | * **-d '' | ||
- | * By default, **find** and **read** delimit their input with newlines; however, since filenames can potentially contain newlines themselves, this default behavior will split up those filenames at the newlines and cause the loop body to fail. | ||
- | * **IFS= **: Set to an empty string, because otherwise read would still strip leading and trailing whitespace. | ||
- | * **|**: | ||
- | * This places the loop in a "sub shell", | ||
- | * To avoid that, you may use a ProcessSubstitution: | ||
- | |||
- | |||
- | |||
- | ---- | ||
- | |||
- | <code bash> | ||
- | while IFS= read -r line; do | ||
- | printf ' | ||
- | done < <(some command) | ||
- | </ | ||
- | |||
- | |||
- | ---- | ||
- | |||
- | ===== My text files are broken! | ||
- | |||
- | If there are some characters after the last line in the file (or to put it differently, | ||
- | |||
- | <code bash> | ||
- | # Emulate cat | ||
- | while IFS= read -r line; do | ||
- | printf ' | ||
- | done < " | ||
- | [[ -n $line ]] && printf %s " | ||
- | </ | ||
- | |||
- | or: | ||
- | |||
- | <code bash> | ||
- | # This does not work: | ||
- | printf 'line 1\ntruncated line 2' | while read -r line; do echo $line; done | ||
- | |||
- | # This does not work either: | ||
- | printf 'line 1\ntruncated line 2' | while read -r line; do echo " | ||
- | |||
- | # This works: | ||
- | printf 'line 1\ntruncated line 2' | { while read -r line; do echo " | ||
- | </ | ||
- | |||
- | The first example, beyond missing the after-loop test, is also missing quotes. See Quotes or Arguments for an explanation why. The Arguments page is an especially important read. | ||
- | |||
- | For a discussion of why the second example above does not work as expected, see FAQ #24. | ||
- | |||
- | ---- | ||
- | |||
- | Alternatively, | ||
- | |||
- | <code bash> | ||
- | while IFS= read -r line || [[ -n $line ]]; do | ||
- | printf ' | ||
- | done < " | ||
- | |||
- | printf 'line 1\ntruncated line 2' | while read -r line || [[ -n $line ]]; do echo " | ||
- | </ | ||
- | |||
- | ---- | ||
- | |||
- | ===== How to keep other commands from " | ||
- | |||
- | Some commands greedily eat up all available data on standard input. | ||
- | |||
- | <code bash> | ||
- | while read -r line; do | ||
- | cat > ignoredfile | ||
- | printf ' | ||
- | done < " | ||
- | </ | ||
- | |||
- | will only print the contents of the first line, with the remaining contents going to " | ||
- | |||
- | ---- | ||
- | |||
- | One workaround is to use a numeric FileDescriptor rather than standard input: | ||
- | |||
- | <code bash> | ||
- | # Bash | ||
- | while IFS= read -r -u 9 line; do | ||
- | cat > ignoredfile | ||
- | printf ' | ||
- | done 9< " | ||
- | |||
- | # Note that read -u is not portable to every shell. Use a redirect to ensure it works in any POSIX compliant shell: | ||
- | while IFS= read -r line <&9; do | ||
- | cat > ignoredfile | ||
- | printf ' | ||
- | done 9< " | ||
- | </ | ||
- | |||
- | or: | ||
- | <code bash> | ||
- | exec 9< " | ||
- | while IFS= read -r line <&9; do | ||
- | cat > ignoredfile | ||
- | printf ' | ||
- | done | ||
- | exec 9<&- | ||
- | </ | ||
- | This example will wait for the user to type something into the file ignoredfile at each iteration instead of eating up the loop input. | ||
- | You might need this, for example, with mencoder which will accept user input if there is any, but will continue silently if there isn' | ||
bash/files/read_a_file.1611669383.txt.gz · Last modified: 2021/01/26 13:56 by peter