Size: 8614
Comment: echo => printf
|
← Revision 56 as of 2025-01-30 04:47:30 ⇥
Size: 8719
Comment: added extra whitespace for breathing room, standardized on 4-space indentation, made "highlight" languages more accurate
|
Deletions are marked like this. | Additions are marked like this. |
Line 8: | Line 8: |
Line 9: | Line 10: |
(( ${#files[*]} )) || echo directory is empty | if (( ${#files[*]} )); then echo 'directory is not empty' else echo 'directory is empty' fi |
Line 23: | Line 30: |
if (shopt -s nullglob dotglob; f=(*); ((! ${#f[@]}))); then echo "The current directory is empty." |
if (shopt -s nullglob dotglob; f=(*); ((${#f[@]}))); then echo 'directory is not empty' else echo 'directory is empty' |
Line 36: | Line 45: |
echo "The current directory contains ${#files[@]} things." | echo "directory contains ${#files[@]} entries" |
Line 44: | Line 53: |
Line 45: | Line 55: |
Line 46: | Line 57: |
echo "The current directory is not empty. It contains:" printf '%s\n' "${files[@]}" |
echo "directory contains ${#files[@]} entries" else echo 'directory is empty' |
Line 58: | Line 70: |
echo "The current directory is not empty." else echo "The current directory is empty." |
echo 'directory is not empty' else echo 'directory is empty' |
Line 71: | Line 83: |
echo "The current directory is not empty." else echo "The current directory is empty." |
echo 'directory is not empty' else echo 'directory is empty' |
Line 81: | Line 93: |
if ( shopt -s dotglob failglob; : ./* ) 2>/dev/null; then echo "The current directory is not empty." else echo "The current directory is empty." |
if (shopt -s dotglob failglob; : ./*) 2>/dev/null; then echo 'directory is not empty' else echo 'directory is empty' |
Line 92: | Line 104: |
Line 93: | Line 106: |
echo "The current directory is not empty." else echo "The current directory is empty." |
echo 'directory is not empty' else echo 'directory is empty' |
Line 102: | Line 115: |
Line 103: | Line 117: |
echo "The current directory is not empty." else echo "The current directory is empty." fi |
echo 'directory is not empty' else echo 'directory is empty' fi |
Line 108: | Line 123: |
Line 109: | Line 125: |
any_match () { local IFS=; { : $@ ;} 2> /dev/null ;} | any_match () { local IFS= { : $@ ;} 2> /dev/null } |
Line 111: | Line 132: |
echo "The current directory is not empty." else echo "The current directory is empty." |
echo 'directory is not empty' else echo 'directory is empty' |
Line 119: | Line 140: |
{{{#!highlight bash | {{{#!highlight sh |
Line 123: | Line 144: |
is_empty=1 |
|
Line 124: | Line 147: |
if test -e "$f" || test -L "$f"; then echo "directory is non-empty" break fi done |
if test -e "$f" || test -L "$f"; then is_empty= break fi done if test "$is_empty"; then echo 'directory is empty' else echo 'directory is not empty' fi |
Line 134: | Line 163: |
{{{#!highlight bash | {{{#!highlight sh |
Line 138: | Line 167: |
if test -e "$f" || test -L "$f"; then n=$((n+1)); fi done printf "There are %d files.\n" "$n" |
if test -e "$f" || test -L "$f"; then n=$((n+1)) fi done printf 'directory contains %d entries\n' "$n" |
Line 145: | Line 176: |
{{{#!highlight bash | {{{#!highlight sh |
Line 150: | Line 181: |
echo "directory is empty" | echo 'directory is empty' else echo 'directory is not empty' |
Line 158: | Line 191: |
{{{#!highlight bash | {{{#!highlight sh |
Line 163: | Line 196: |
printf "There are %d files.\n" "$n" | printf 'directory contains %d files\n' "$n" |
Line 172: | Line 205: |
{{{#!highlight bash | {{{#!highlight sh |
Line 175: | Line 208: |
find "$somedir" -maxdepth 0 -empty -exec printf '%s is empty.\n' {} \; # GNU/BSD | find "$somedir" -prune -empty -exec printf '%s is empty\n' {} \; # GNU/BSD |
Line 181: | Line 214: |
{{{#!highlight bash | {{{#!highlight sh |
Line 193: | Line 226: |
{{{#!highlight bash | {{{#!highlight ksh |
How can I check whether a directory is empty or not? How do I check for any *.mpg files, or count how many there are?
In Bash, you can count files safely and easily with the nullglob and dotglob options (which change the behaviour of globbing), and an array:
See ArithmeticExpression for explanations of arithmetic commands.
Of course, you can use any glob you like instead of *. E.g. *.mpg or /my/music/*.mpg works fine.
Bear in mind that you need read permission on the directory, or it will always appear empty.
Some people dislike nullglob because having unmatched globs vanish altogether confuses programs like ls. Mistyping ls *.zip as ls *.zpi may cause every file to be displayed (for such cases consider setting failglob). Setting nullglob in a SubShell avoids accidentally changing its setting in the rest of the shell, at the price of an extra fork(). If you'd like to avoid having to set and unset shell options, you can pour it all into a SubShell:
The other disadvantage of this approach (besides the extra fork()) is that the array is lost when the subshell exits. If you planned to use those filenames later, then they have to be retrieved all over again.
Both of these examples expand a glob and store the resulting filenames into an array, and then check whether the number of elements in the array is 0. If you actually want to see how many files there are, just print the array's size instead of checking whether it's 0:
You can also avoid the nullglob if you're OK with putting a non-existing filename in the array should no files match (instead of an empty array):
Without nullglob, if there are no files in the directory, the glob will be added as the only element in the array. Since * is a valid filename, we can't simply check whether the array contains a literal *. So instead, we check whether the thing in the array exists as a file. The -L test is required because -e fails if the first file is a dangling symlink.
If you don't care how many matching files there are and don't want to store the results in an array, you can use bash's compgen command. Unfortunately, due to a bug, you need to use a hack to make it recognize dotglob:
Or you can use an extended glob:
You may also use failglob:
But, if you use failglob, note that the subshell is required; the following code does not work because failglob will raise a shell error that will cause bash to stop running the current command (including the if command, any outer compound command, and the entire function that ran this code if it is part of a function), so this will only work in the true case, the else branch will never run:
If you really want to avoid using the subshell and want to set failglob globally, you can either "catch" the shell error using command eval, or you can write a function that expands the glob indirectly:
1 shopt -s dotglob failglob
2
3 if command eval ': ./*' 2> /dev/null; then
4 echo 'directory is not empty'
5 else
6 echo 'directory is empty'
7 fi
8
9 # or
10
11 shopt -s dotglob failglob
12
13 any_match () {
14 local IFS=
15 { : $@ ;} 2> /dev/null
16 }
17
18 if any_match './*'; then
19 echo 'directory is not empty'
20 else
21 echo 'directory is empty'
22 fi
If your script needs to run with various non-Bash shell implementations, you can try using an external program like python, perl, or find; or you can try one of these. Note the "magic 3 globs"1 as POSIX does not have the dotglob option.
1 # POSIX
2 # Clobbers the positional parameters, so make sure you don't need them.
3 set -- * .[!.]* ..?*
4
5 is_empty=1
6 for f in "$@"; do
7 if test -e "$f" || test -L "$f"; then
8 is_empty=
9 break
10 fi
11 done
12
13 if test "$is_empty"; then
14 echo 'directory is empty'
15 else
16 echo 'directory is not empty'
17 fi
At this stage, the positional parameters have been loaded with the contents of the directory, and can be used for processing.
If you just want to count files:
In the Bourne shell, it's even worse, because there is no test -e or test -L:
Of course, that fails if * exists as something other than a plain file (such as a directory or FIFO). The absence of a -e test really hurts.
Here is another solution using find:
If you want it not to recurse, then you need to tell find not to recurse into directories. This gets really tricky and ugly. GNU find has a -maxdepth option to do it. With standard POSIX find, you're stuck with -prune. This is left as an exercise for the reader.
Never try to parse ls output. Even ls -A solutions can break (e.g. on HP-UX, if you are root, ls -A does the exact opposite of what it does if you're not root -- and no, I can't make up something that incredibly stupid).
In fact, one may wish to avoid the direct question altogether. Usually people want to know whether a directory is empty because they want to do something involving the files therein, etc. Look to the larger question. For example, one of these find-based examples may be an appropriate solution:
Most commonly, all that's really needed is something like this:
In other words, the person asking the question may have thought an explicit empty-directory test was needed to avoid an error message like mympgviewer: ./*.mpg: No such file or directory when in fact no such test is required.
Support for a nullglob-like feature is inconsistent. In ksh93 it can be done on a per-pattern basis by prefixing with ~(N)2:
From: http://permalink.gmane.org/gmane.comp.standards.posix.austin.general/2058, which contains some good discussion. (2)