mirror of
https://github.com/rawiriblundell/wiki.bash-hackers.org
synced 2024-11-02 00:53:07 +01:00
180 lines
6.2 KiB
Markdown
180 lines
6.2 KiB
Markdown
# The unset builtin command
|
|
|
|
## Synopsis
|
|
|
|
unset [-f|v] [-n] [NAME ...]
|
|
|
|
## Description
|
|
|
|
The `unset` builtin command is used to unset values and attributes of
|
|
shell variables and functions. Without any option, `unset` tries to
|
|
unset a variable first, then a function.
|
|
|
|
### Options
|
|
|
|
| Option | Description |
|
|
|:-------|:-------------------------------------------------------------------------------------------------------------|
|
|
| `-f` | treats each `NAME` as a function name |
|
|
| `-v` | treats each `NAME` as a variable name |
|
|
| `-n` | treats each `NAME` as a name reference and unsets the variable itself rather than the variable it references |
|
|
|
|
### Exit status
|
|
|
|
| Status | Reason |
|
|
|:-------|:---------------------------------------------------|
|
|
| 0 | no error |
|
|
| !=0 | invalid option |
|
|
| !=0 | invalid combination of options (`-v` **and** `-f`) |
|
|
| !=0 | a given `NAME` is read-only |
|
|
|
|
## Examples
|
|
|
|
unset -v EDITOR
|
|
|
|
unset -f myfunc1 myfunc2
|
|
|
|
### Scope
|
|
|
|
In bash, unset has some interesting properties due to its unique dynamic
|
|
scope. If a local variable is both declared and unset (by calling unset
|
|
on the local) from within the same function scope, then the variable
|
|
appears unset to that scope and all child scopes until either returning
|
|
from the function, or another local variable of the same name is
|
|
declared underneath where the original variable was unset. In other
|
|
words, the variable looks unset to everything until returning from the
|
|
function in which the variable was set (and unset), at which point
|
|
variables of the same name from higher scopes are uncovered and
|
|
accessible once again.
|
|
|
|
If however unset is called from a child scope relative to where a local
|
|
variable has been set, then the variable of the same name in the
|
|
next-outermost scope becomes visible to its scope and all children - as
|
|
if the variable that was unset was never set to begin with. This
|
|
property allows looking upwards through the stack as variable names are
|
|
unset, so long as unset and the local it unsets aren't together in the
|
|
same scope level.
|
|
|
|
Here's a demonstration of this behavior.
|
|
|
|
#!/usr/bin/env bash
|
|
|
|
FUNCNEST=10
|
|
|
|
# Direct recursion depth.
|
|
# Search up the stack for the first non-FUNCNAME[1] and count how deep we are.
|
|
callDepth() {
|
|
# Strip "main" off the end of FUNCNAME[@] if current function is named "main" and
|
|
# Bash added an extra "main" for non-interactive scripts.
|
|
if [[ main == !(!("${FUNCNAME[1]}")|!("${FUNCNAME[-1]}")) && $- != *i* ]]; then
|
|
local -a 'fnames=("${FUNCNAME[@]:1:${#FUNCNAME[@]}-2}")'
|
|
else
|
|
local -a 'fnames=("${FUNCNAME[@]:1}")'
|
|
fi
|
|
|
|
if (( ! ${#fnames[@]} )); then
|
|
printf 0
|
|
return
|
|
fi
|
|
|
|
local n
|
|
while [[ $fnames == ${fnames[++n]} ]]; do
|
|
:
|
|
done
|
|
|
|
printf -- $n
|
|
}
|
|
|
|
# This function is the magic stack walker.
|
|
unset2() {
|
|
unset -v -- "$@"
|
|
}
|
|
|
|
f() {
|
|
local a
|
|
if (( (a=$(callDepth)) <= 4 )); then
|
|
(( a == 1 )) && unset a
|
|
(( a == 2 )) && declare -g a='global scope yo'
|
|
f
|
|
else
|
|
trap 'declare -p a' DEBUG
|
|
unset2 a # declare -- a="5"
|
|
unset a a # declare -- a="4"
|
|
unset a # declare -- a="2"
|
|
unset a # ./unset-tests: line 44: declare: a: not found
|
|
: # declare -- a="global scope yo"
|
|
fi
|
|
}
|
|
|
|
a='global scope'
|
|
f
|
|
|
|
# vim: set fenc=utf-8 ff=unix ts=4 sts=4 sw=4 ft=sh nowrap et:
|
|
|
|
output:
|
|
|
|
declare -- a="5"
|
|
declare -- a="4"
|
|
declare -- a="2"
|
|
./unset-tests: line 44: declare: a: not found
|
|
declare -- a="global scope yo"
|
|
|
|
Some things to observe:
|
|
|
|
- `unset2` is only really needed once. We remain 5 levels deep in `f`'s
|
|
for the remaining `unset` calls, which peel away the outer layers of
|
|
`a`'s.
|
|
- Notice that the "a" is unset using an ordinary unset command at
|
|
recursion depth 1, and subsequently calling unset reveals a again in
|
|
the global scope, which has since been modified in a lower scope using
|
|
declare -g.
|
|
- Declaring a global with declare -g bypasses all locals and sets or
|
|
modifies the variable of the global scope (outside of all functions).
|
|
It has no affect on the visibility of the global.
|
|
- This doesn't apply to individual array elements. If two local arrays
|
|
of the same name appear in different scopes, the entire array of the
|
|
inner scope needs to be unset before any elements of the outer array
|
|
become visible. This makes "unset" and "unset2" identical for
|
|
individual array elements, and for arrays as a whole, unset and unset2
|
|
behave as they do for scalar variables.
|
|
|
|
### Args
|
|
|
|
Like several other Bash builtins that take parameter names, unset
|
|
expands its arguments.
|
|
|
|
~ $ ( a=({a..d}); unset 'a[2]'; declare -p a )
|
|
declare -a a='([0]="a" [1]="b" [3]="d")'
|
|
|
|
As usual in such cases, it's important to quote the args to avoid
|
|
accidental results such as globbing.
|
|
|
|
~ $ ( a=({a..d}) b=a c=d d=1; set -x; unset "${b}["{2..3}-c\]; declare -p a )
|
|
+ unset 'a[2-1]' 'a[3-1]'
|
|
+ declare -p a
|
|
declare -a a='([0]="a" [3]="d")'
|
|
|
|
Of course hard to follow indirection is still possible whenever
|
|
arithmetic is involved, also as shown above, even without extra
|
|
expansions.
|
|
|
|
In Bash, the `unset` builtin only evaluates array subscripts if the
|
|
array itself is set.
|
|
|
|
~ $ ( unset -v 'a[$(echo a was set >&2)0]' )
|
|
~ $ ( a=(); unset -v 'a[$(echo a was set >&2)0]' )
|
|
a was set
|
|
|
|
## Portability considerations
|
|
|
|
Quoting POSIX:
|
|
|
|
If neither -f nor -v is specified, name refers to a variable; if a variable by that name does not exist, it is unspecified whether a function by that name, if any, shall be unset.
|
|
|
|
Therefore, it is recommended to explicitly specify `-f` or `-v` when
|
|
using `unset`. Also, I prefer it as a matter of style.
|
|
|
|
## See also
|
|
|
|
- [declare](/commands/builtin/declare)
|
|
- [unset](/commands/builtin/unset)
|