Shell如何检查一个目录是否为空?
wptr33 2024-12-10 21:20 17 浏览
如何检查一个目录是否为空?如何检查是否存在任何 *.mpg 文件,或者计算有多少个这样的文件?
在 Bash 中,你可以通过使用 nullglob 和 dotglob 选项(这会改变 globbing 的行为),以及一个 array,安全而简便地计算文件数量:
# Bash
shopt -s nullglob dotglob
files=(*)
(( ${#files[*]} )) || echo 目录为空
shopt -u nullglob dotglob
当然,你可以使用任何你喜欢的 glob,而不仅仅是 。比如,.mpg 或者 /my/music/*.mpg 都可以正常工作。
请注意,你需要对目录拥有读取权限,否则它将始终显示为空。
一些人不喜欢 nullglob,因为没有匹配的 glob 会完全消失,这会让像 ls 这样的程序感到困惑。如果将 ls *.zip 错误地输入为 ls *.zpi,可能会导致显示所有文件(对于这种情况,考虑设置 failglob)。在子 shell 中设置 nullglob 可以避免意外地改变其余部分的 shell 设置,但代价是额外的 fork()。如果你想避免设置和取消设置 shell 选项,你可以将所有内容放入一个子 shell:
# Bash
if (shopt -s nullglob dotglob; f=(*); ((! ${#f[@]}))); then
echo "当前目录为空。"
fi
这种方法的另一个缺点(除了额外的 fork())是,当子 shell 退出时,数组会丢失。如果你计划稍后使用这些文件名,那么它们必须再次检索。
这两个示例都会扩展 glob 并将结果文件名存储到一个数组中,然后检查数组中元素的数量是否为 0。如果你实际上想要查看有多少个文件,只需打印数组的大小,而不是检查它是否为 0:
# Bash
shopt -s nullglob dotglob
files=(*)
echo "当前目录包含 ${#files[@]} 个文件。"
如果你不介意在数组中放入一个不存在的文件名(而不是一个空数组),你也可以避免使用 nullglob:
# Bash
shopt -s dotglob
files=(*)
if [[ -e ${files[0]} || -L ${files[0]} ]]; then
echo "当前目录不为空。它包含以下文件:"
printf '%s\n' "${files[@]}"
fi
如果目录中没有文件,则没有 nullglob,这个 glob 将被添加为数组中的唯一元素。由于 * 是一个有效的文件名,我们无法简单地检查数组是否包含一个字面上的 *。因此,我们检查数组中的内容是否作为文件存在。需要注意的是,-L 测试是必需的,因为 -e 如果第一个文件是一个 dangling symlink,将失败。
如果您不关心有多少匹配的文件,并且不想将结果存储在数组中,您可以使用Bash的compgen命令。不幸的是,由于一个bug,您需要使用一个技巧让其识别dotglob:
# Bash
if (shopt -s dotglob; : *; compgen -G '*' >/dev/null); then
echo "当前目录不为空。"
else
echo "当前目录为空。"
fi
或者您可以使用扩展的glob:
# Bash
# 可以通过为整个脚本启用extglob来避免子shell。
# 这样做是安全的。
if (shopt -s extglob; compgen -G '@(*|.[!.]*|..?*)' >/dev/null); then
echo "当前目录不为空。"
else
echo "当前目录为空。"
fi
您还可以使用failglob:
# Bash
if (shopt -s dotglob failglob; : ./*) 2>/dev/null; then
echo "当前目录不为空。"
else
echo "当前目录为空。"
fi
但是,请注意,如果您使用failglob,则需要子shell;下面的代码无法运行,因为failglob会引发一个shell错误,导致Bash停止运行当前命令(包括if命令,任何外部复合命令以及运行此代码的整个函数,如果它是函数的一部分),因此这仅适用于true的情况,else分支永远不会运行:
# 错误的代码!
shopt -s dotglob failglob
if { : ./*; } 2>/dev/null; then
echo "当前目录不为空。"
else
echo "当前目录为空。"
fi
如果您真的想避免使用子shell并且想要全局设置failglob,您可以使用命令eval来“捕获”shell错误,或者您可以编写一个间接展开glob的函数:
shopt -s dotglob failglob
if command eval ': ./*' 2>/dev/null; then
echo "当前目录不为空。"
else
echo "当前目录为空。"
fi
# 或者
shopt -s dotglob failglob
any_match () { local IFS=; { : "$@"; } 2>/dev/null; }
if any_match './*'; then
echo "当前目录不为空。"
else
echo "当前目录为空。"
fi
如果您的脚本需要在各种非Bash shell实现中运行,您可以尝试使用外部程序如Python、Perl或find;或者您可以尝试以下方法。请注意“magic 3 globs”^[1],因为POSIX没有dotglob选项。
# POSIX
# 这会破坏位置参数,所以确保您不需要它们。
set -- * .[!.]* ..?*
for f in "$@"; do
if test -e "$f" || test -L "$f"; then
echo "目录不为空"
break
fi
done
在这个阶段,位置参数已经加载了目录的内容,并且可以用于处理。
如果你只想计算文件数量:
# POSIX
n=0
for f in * .[!.]* ..?*; do
if test -e "$f" || test -L "$f"; then n=$((n+1)); fi
done
printf "共有 %d 个文件。\n" "$n"
在 Bourne shell 中,情况更糟糕,因为没有 test -e 或 test -L?:
# Bourne
# (当然,系统必须有 printf(1)。)
if test "`printf '%s %s %s' .* *`" = '. .. *' && test ! -f '*'
then
echo "目录为空"
fi
当然,如果 *? 存在于普通文件以外的其他内容(如目录或 FIFO),这种方法会失败。缺少 -e? 测试真的很痛苦。
这里还有另一种使用 find? 的解决方案:
# POSIX
# 打印每个文件的一个 `.` 并计算打印的字符数。
# 这个解决方案将递归。如果不想递归,请参见下面的内容。
n=$(find . -type f -exec printf %.0s. {} + | wc -m)
printf "共有 %d 个文件。\n" "$n"
如果你不希望递归,那么你需要告诉 find? 不要递归进入目录。这变得非常棘手和丑陋。GNU find? 有一个 -maxdepth? 选项来实现此功能。对于标准的 POSIX find?,你只能使用 -prune?。这留给读者作为练习。
永远不要尝试解析 ls? 的输出。即使是 ls -A? 的解决方案也可能会出错(例如,在 HP-UX 上,如果你是 root 用户,ls -A? 与非 root 用户的行为正好相反--不,我不能编造出这么愚蠢的东西)。
实际上,你可能希望完全避免直接的 *问题。通常人们想知道一个目录是否为空是因为他们想在其中的文件中进行一些操作等。要看到更大的问题。例如,这些基于 find? 的例子可能是一个合适的解决方案:
# Bourne / POSIX
find "$somedir" -type f -exec echo 发现意外的文件 {} \;
find "$somedir" -prune -empty -exec printf '%s 是空的。\n' {} \; # GNU/BSD
find "$somedir" -type d -empty -exec cp /my/configfile {} \; # GNU/BSD
最常见的情况是只需要像这样:
# Bourne / POSIX
for f in ./*.mpg; do
test -f "$f" || continue
mympgviewer "$f"
done
换句话说,提问者可能认为需要显式的空目录测试来避免出现错误消息,比如 mympgviewer: ./*.mpg: No such file or directory?,当实际上并不需要这样的测试。
对于类似 nullglob 的特性的支持是不一致的。在 ksh93 中,可以通过在模式前加上 ~(N)? 来实现每个模式的基础:
# ksh93
for f in ~(N)*; do
....
done
如果您觉得文章内容对你有一点帮助可以关注我,我在头条平台会持续分享更多实用的shell技巧和最佳实践,如果想系统的快速学习shell的各种高阶用法和生产环境避坑指南可以看看《shell脚本编程最佳实践》专栏,专栏里有更多的实用小技巧和脚本代码分享。
- 上一篇:用bash脚本自动下载ftp服务器文件
- 下一篇:Shell脚本编程实战
相关推荐
- 「网络安全」JAVA代码审计——XXE外部实体注入
-
一、WEB安全部分想要了解XXE,在那之前需要了解XML的相关基础二、XML基础...
- Web前端面试题目及答案汇总(web前端面试题最新)
-
Web前端面试题目及答案汇总来源:极客头条以下是收集一些面试中经常会遇到的经典面试题以及自己面试过程中无法解决的问题,通过对知识的整理以及经验的总结,重新巩固自身的前端基础知识,如有错误或更好的答案,...
- 什么是脚本文件?与可执行文件有什么不同?
-
今天的内容是脚本文件和可执行文件是两种不同类型的计算机文件,它们在结构和执行方式上有显著区别。脚本文件:定义与特性...
- 20个实用Python运维脚本(收藏级)(python 运维工具)
-
系统环境:支持Linux(Ubuntu/CentOS/Debian)和Windows...
- 2026年前每个开发者都应该学习的技能
-
优秀开发者...
- Linux 如何每 5、10、15 或 30 分钟运行一次 Cron 作业?
-
在Linux系统中,Cron是一个强大的工具,用于自动化重复性任务。通过合理配置...
- Shell脚本编程进阶实战:从入门到高效自动化
-
Shell脚本编程进阶实战:从入门到高效自动化一、参数处理进阶:打造专业级CLI工具1.高级参数解析示例...
- 在Bash中按分隔符拆分字符串的方法
-
技术背景在Bash脚本编程中,经常会遇到需要按特定分隔符拆分字符串的需求,例如处理CSV文件、解析日志等。掌握字符串拆分的方法对于数据处理和脚本自动化非常重要。...
- 程序员用5分钟,把一个400多MB的苹果安装包削掉了187MB
-
丰色发自凹非寺量子位|公众号QbitAI前些日子,一个...
- 如何在 Windows 上编写批处理脚本
-
你知道如何使用命令提示符吗?如果这样做,您可以编写一个批处理文件。在最简单的形式中,批处理文件(或批处理脚本)是双击文件时执行的几个命令的列表。批处理文件一直回到DOS,但仍然适用于现代版本的Win...
- 一文搞懂shell脚本(shell脚本应用实战)
-
一文搞懂shell脚本1、shell脚本介绍什么是shell脚本...
- 一文讲清ShellScript脚本编程知识
-
摘要:本文详尽地讲述了ShellScript的基础内容,还有它在Linux系统里的运用情况,涵盖了它的基本语法、常用的命令以及高级的功能。ShellScript可是一种简单又非常实用的编...
- 在Bash脚本中获取自身所在目录的方法
-
技术背景在使用Bash脚本时,有时需要获取脚本自身所在的目录。比如,当脚本作为另一个应用程序的启动器时,需要将工作目录更改为脚本所在的目录,以便对该目录中的文件进行操作。然而,由于脚本的调用方式多样(...
- shell中如何确定脚本的位置?这篇文章告诉你
-
我想从同一个位置读取一些配置文件,如何确定脚本的位置?。这个问题的出现主要是由两个原因引发的:一是您希望将脚本的数据或配置进行外部化,因此需要一种方式来寻找这些外部资源;二是您的脚本需要对某些捆绑资源...
- bash shell 语法(bash命令用法)
-
下面是**Shell(Bash)语法的常用知识点总结**,适合初学者和日常脚本编写参考。内容涵盖变量、判断、循环、函数、重定向、正则、数组等常见用法。---#Shell(Bash)语法速查总结...
- 一周热门
-
-
C# 13 和 .NET 9 全知道 :13 使用 ASP.NET Core 构建网站 (1)
-
因果推断Matching方式实现代码 因果推断模型
-
git pull命令使用实例 git pull--rebase
-
面试官:git pull是哪两个指令的组合?
-
git 执行pull错误如何撤销 git pull fail
-
git pull 和git fetch 命令分别有什么作用?二者有什么区别?
-
git fetch 和git pull 的异同 git中fetch和pull的区别
-
git pull 之后本地代码被覆盖 解决方案
-
还可以这样玩?Git基本原理及各种骚操作,涨知识了
-
git命令之pull git.pull
-
- 最近发表
- 标签列表
-
- git pull (33)
- git fetch (35)
- mysql insert (35)
- mysql distinct (37)
- concat_ws (36)
- java continue (36)
- jenkins官网 (37)
- mysql 子查询 (37)
- python元组 (33)
- mybatis 分页 (35)
- vba split (37)
- redis watch (34)
- python list sort (37)
- nvarchar2 (34)
- mysql not null (36)
- hmset (35)
- python telnet (35)
- python readlines() 方法 (36)
- munmap (35)
- docker network create (35)
- redis 集合 (37)
- python sftp (37)
- setpriority (34)
- c语言 switch (34)
- git commit (34)