Size: 2285
Comment: math context for trimming leading zeroes, etc.
|
Size: 3147
Comment:
|
Deletions are marked like this. | Additions are marked like this. |
Line 2: | Line 2: |
Line 5: | Line 6: |
Here's one that only works for whitespace. It relies on the fact that `read` strips all leading and trailing whitespace when `IFS` isn't set: | For simple variables, you can trim whitespace (or other characters) using this trick: {{{ # POSIX junk="${var##*[! ]}" # match prefixed spaces var="${var%"$junk"}" # trim the match junk="${var%%[! ]*}" # match affixed spaces var="${var#"$junk"}" # trim the match }}} Bash can do the same thing, but without the need for a throw-away variable, by using [[glob|extglob]]'s more advanced pattern matching: {{{ # Bash shopt -s extglob var="${var##*( )}" # trim the left var="${var%%*( )}" # trim the right }}} Here's one that only works for whitespace. It relies on the fact that `read` strips all leading and trailing whitespace (tab or space character) when `IFS` isn't set: |
Line 12: | Line 31: |
Bash can do something similar with a "here string": | |
Line 13: | Line 33: |
Bash can do something similar with a "here string": | |
Line 18: | Line 37: |
Using an empty string as a delimiter means the read consumes the whole string, as NUL is used. (Remember: BASH only does C-string variables.) This is entirely safe for any text, including newlines. | Using an empty string as a delimiter means the read consumes the whole string, as NUL is used. (Remember: BASH only does C-string variables.) This is entirely safe for any text, including newlines (which will also be stripped from the beginning and end of the variable with the default value of IFS). |
Line 21: | Line 40: |
Line 25: | Line 45: |
shopt -u extglob | |
Line 27: | Line 46: |
(where `[[:space:]]` includes space, tab and all other horizontal and vertical spacing characters, the list of which varies with the current [[locale]]). | |
Line 29: | Line 49: |
Line 33: | Line 54: |
This solution isn't restricted to whitespace like the first few were. You can remove leading zeroes as well: | |
Line 34: | Line 56: |
This solution isn't restricted to whitespace like the first few were. You can remove leading zeroes as well: | |
Line 40: | Line 61: |
Another way to remove leading zeroes from a number in bash is to treat it as a decimal integer, in a [[ArithmeticExpression|math context]]: | |
Line 41: | Line 63: |
Another way to remove leading zeroes from a number in bash is to treat it as an integer, in a [[ArithmeticExpression|math context]]: | |
Line 45: | Line 66: |
# However, this fails if x contains anything other than digits. | # However, this fails if x contains anything other than decimal digits. |
Line 47: | Line 68: |
If you need to remove leading zeroes in a POSIX shell, you can use a loop: | |
Line 48: | Line 70: |
If you need to remove leading zeroes in a POSIX shell, you can use a loop: | |
Line 58: | Line 79: |
Or this trick (covered in more detail in [[BashFAQ/100|FAQ #100]]): | |
Line 59: | Line 81: |
Or this trick (covered in more detail in [[BashFAQ/100|FAQ #100]]): | |
Line 63: | Line 84: |
var=${var#$zeroes} | var=${var#"$zeroes"} |
Line 65: | Line 86: |
There are many, many other ways to do this, using sed for instance: | |
Line 66: | Line 88: |
There are many, many other ways to do this, using sed for instance: | |
Line 69: | Line 90: |
x=$(echo "$x" | sed -e 's/^[[:space:]]*//' -e 's/[[:space:]]*$//') | x=$(printf '%s\n' "$x" | sed -e 's/^[[:space:]]*//' -e 's/[[:space:]]*$//') |
How can I trim leading/trailing white space from one of my variables?
There are a few ways to do this. Some involve special tricks that only work with whitespace. Others are more general, and can be used to strip leading zeroes, etc.
For simple variables, you can trim whitespace (or other characters) using this trick:
# POSIX junk="${var##*[! ]}" # match prefixed spaces var="${var%"$junk"}" # trim the match junk="${var%%[! ]*}" # match affixed spaces var="${var#"$junk"}" # trim the match
Bash can do the same thing, but without the need for a throw-away variable, by using extglob's more advanced pattern matching:
# Bash shopt -s extglob var="${var##*( )}" # trim the left var="${var%%*( )}" # trim the right
Here's one that only works for whitespace. It relies on the fact that read strips all leading and trailing whitespace (tab or space character) when IFS isn't set:
# POSIX, but fails if the variable contains newlines read -r var << EOF $var EOF
Bash can do something similar with a "here string":
# Bash read -rd '' x <<< "$x"
Using an empty string as a delimiter means the read consumes the whole string, as NUL is used. (Remember: BASH only does C-string variables.) This is entirely safe for any text, including newlines (which will also be stripped from the beginning and end of the variable with the default value of IFS).
Here's a solution using extglob together with parameter expansion:
# Bash shopt -s extglob x=${x##+([[:space:]])} x=${x%%+([[:space:]])}
(where [[:space:]] includes space, tab and all other horizontal and vertical spacing characters, the list of which varies with the current locale).
This also works in KornShell, without needing the explicit extglob setting:
# ksh x=${x##+([[:space:]])} x=${x%%+([[:space:]])}
This solution isn't restricted to whitespace like the first few were. You can remove leading zeroes as well:
# Bash shopt -s extglob x=${x##+(0)}
Another way to remove leading zeroes from a number in bash is to treat it as a decimal integer, in a math context:
# Bash x=$((10#$x)) # However, this fails if x contains anything other than decimal digits.
If you need to remove leading zeroes in a POSIX shell, you can use a loop:
# POSIX while true; do case "$var" in 0*) var=${var#0};; *) break;; esac done
Or this trick (covered in more detail in FAQ #100):
# POSIX zeroes=${var%%[!0]*} var=${var#"$zeroes"}
There are many, many other ways to do this, using sed for instance:
# POSIX, suppress the trailing and leading whitespace on every line x=$(printf '%s\n' "$x" | sed -e 's/^[[:space:]]*//' -e 's/[[:space:]]*$//')
Solutions based on external programs like sed are better suited to trimming large files, rather than shell variables.