02. Shell 工具和脚本
练习 1
阅读
man ls,然后使用ls命令进行如下操作:
- 所有文件(包括隐藏文件)
- 文件打印以人类可以理解的格式输出 (例如,使用 454M 而不是 454279954)
- 文件以最近访问顺序排序
- 以彩色文本显示输出结果
典型输出如下:
-rw-r--r-- 1 user group 1.1M Jan 14 09:53 baz drwxr-xr-x 5 user group 160 Jan 14 09:53 . -rw-r--r-- 1 user group 514 Jan 14 06:42 bar -rw-r--r-- 1 user group 106M Jan 13 12:12 foo drwx------+ 47 user group 1.5K Jan 12 18:08 ..
ls -laht --color=auto练习 2
编写两个 bash 函数
marco和polo执行下面的操作。每当你执行marco时,当前的工作目录应当以某种形式保存,当执行polo时,无论现在处在什么目录下,都应当cd回到当时执行marco的目录。为了方便 debug,你可以把代码写在单独的文件marco.sh中,并通过source marco.sh命令,(重新)加载函数。
#!/usr/bin/env bash
macro() {
export MACRO_DIR="$PWD"
echo "当前工作目录已保存为 MACRO_DIR: $MACRO_DIR"
}
polo() {
if [[ -z "$MACRO_DIR" ]]; then
echo "未找到保存的工作目录,请先执行 macro 命令"
return 1
fi
if cd "$MACRO_DIR"; then
echo "已返回到保存的工作目录: $MACRO_DIR"
else
echo "无法切换到保存的工作目录: $MACRO_DIR"
return 1
fi
}练习 3
假设您有一个命令,它很少出错。因此为了在出错时能够对其进行调试,需要花费大量的时间重现错误并捕获输出。编写一段 bash 脚本,运行如下的脚本直到它出错,将它的标准输出和标准错误流记录到文件,并在最后输出所有内容。加分项:报告脚本在失败前共运行了多少次。
sh#!/usr/bin/env bash n=$(( RANDOM % 100 )) if [[ n -eq 42 ]]; then echo "Something went wrong" >&2 echo "The error was using magic numbers" exit 1 fi echo "Everything went according to plan"
#!/usr/bin/env bash
count=0
echo > out.log
while true;do
./buggy.sh > out.log 2>&1
if [[ $? -ne 0 ]]; then
break
fi
((count++))
done
echo "在成功运行 $count 次后失败"
cat out.log练习 4
本节课我们讲解的
find命令中的-exec参数非常强大,它可以对我们查找的文件进行操作。但是,如果我们要对所有文件进行操作呢?例如创建一个 zip 压缩文件?我们已经知道,命令行可以从参数或标准输入接受输入。在用管道连接命令时,我们将标准输出和标准输入连接起来,但是有些命令,例如tar则需要从参数接受输入。这里我们可以使用xargs命令,它可以使用标准输入中的内容作为参数。 例如ls | xargs rm会删除当前目录中的所有文件。您的任务是编写一个命令,它可以递归地查找文件夹中所有的 HTML 文件,并将它们压缩成 zip 文件。注意,即使文件名中包含空格,您的命令也应该能够正确执行(提示:查看
xargs的参数-d,译注:MacOS 上的xargs没有-d,查看这个 issue)如果您使用的是 MacOS,请注意默认的 BSD
find与 GNU coreutils 中的是不一样的。你可以为find添加-print0选项,并为xargs添加-0选项。作为 Mac 用户,您需要注意 mac 系统自带的命令行工具和 GNU 中对应的工具是有区别的;如果你想使用 GNU 版本的工具,也可以使用 brew 来安装。
练习 5
(进阶)编写一个命令或脚本递归的查找文件夹中最近修改的文件。更通用的做法,你可以按照最近的修改时间列出文件吗?