gpt4 book ai didi

bash - 在 bash 脚本中创建 which 命令

转载 作者:行者123 更新时间:2023-12-04 14:33:18 24 4
gpt4 key购买 nike

对于一项作业,我应该创建一个名为 my_which.sh 的脚本,它将“做与 Unix 命令相同的事情,但使用 for 循环而不是 if”。我也不允许在我的脚本中调用 which。

我对此是全新的,并且一直在阅读教程,但我对如何开始感到很困惑。 which不就是列出命令的路径名吗?

如果是这样,我将如何在不调用 which 的情况下使用 for 循环和 if 语句来显示正确的路径名?

例如,如果我运行我的脚本,它将echo % 并等待输入。但是,我该如何将其转化为查找目录呢?所以它看起来像这样?

#!/bin/bash
path=(`echo $PATH`)
echo -n "% "
read ans
for i in $path
do
if [ -d $i ]; then
echo $i
fi
done

我将不胜感激任何帮助,甚至任何可以帮助我开始这方面的入门教程。老实说,我很困惑应该如何实现。

最佳答案

  1. 安全地拆分您的PATH 变量。这是在分隔符处拆分字符串的通用方法,对于任何可能的字符(包括换行符)都是 100% 安全的:

    IFS=: read -r -d '' -a paths < <(printf '%s:\0' "$PATH")

    我们人为添加了:,因为如果PATH以尾随:结尾,则可以理解当前目录应该在中路径。虽然这是危险的并且不推荐,但如果我们想模仿 which,我们也必须考虑到这一点。如果没有这个尾随冒号,PATH/bin:/usr/bin: 将被分成

    declare -a paths='( [0]="/bin" [1]="/usr/bin" )'

    而尾随冒号的结果数组是:

    declare -a paths='( [0]="/bin" [1]="/usr/bin" [2]="" )'

    这是其他答案遗漏的一个细节。当然,只有当 PATH 已设置且非空时,我们才会这样做。

  2. 通过拆分 PATH,我们将使用 for 循环来检查是否可以在给定目录中找到参数。请注意,仅当参数不包含 / 字符时才应执行此操作!这也是其他答案遗漏的东西。

  3. 我的版本处理一个独特的选项-a打印每个参数的所有匹配路径名。否则,只打印第一个匹配项。我们也必须考虑到这一点。

  4. 我的版本处理以下退出状态:

       0      if all specified commands are found and executable

    1 if one or more specified commands is nonexistent or not executable

    2 if an invalid option is specified

    我们也会处理。

我猜下面的代码相当忠实地模仿了我的 which 的行为(而且它是纯 Bash):

#!/bin/bash

show_usage() {
printf 'Usage: %s [-a] args\n' "$0"
}

illegal_option() {
printf >&2 'Illegal option -%s\n' "$1"
show_usage
exit 2
}

check_arg() {
if [[ -f $1 && -x $1 ]]; then
printf '%s\n' "$1"
return 0
else
return 1
fi
}

# manage options

show_only_one=true

while (($#)); do
[[ $1 = -- ]] && { shift; break; }
[[ $1 = -?* ]] || break
opt=${1#-}
while [[ $opt ]]; do
case $opt in
(a*) show_only_one=false; opt=${opt#?} ;;
(*) illegal_option "${opt:0:1}" ;;
esac
done
shift
done

# If no arguments left or empty PATH, exit with return code 1
(($#)) || exit 1
[[ $PATH ]] || exit 1

# split path
IFS=: read -r -d '' -a paths < <(printf '%s:\0' "$PATH")

ret=0
# loop on arguments
for arg; do
# Check whether arg contains a slash
if [[ $arg = */* ]]; then
check_arg "$arg" || ret=1
else
this_ret=1
for p in "${paths[@]}"; do
if check_arg "${p:-.}/$arg"; then
this_ret=0
"$show_only_one" && break
fi
done
((this_ret==1)) && ret=1
fi
done

exit "$ret"

为了测试一个参数是否可执行,我正在检查它是否是一个常规文件1,它是可执行的:

[[ -f $arg && -x $arg ]]

我猜这与我的 which 的行为很接近。


1 正如@mklement0 指出的(谢谢!)-f 测试,当应用于符号链接(symbolic link)时,测试符号链接(symbolic link)的目标.

关于bash - 在 bash 脚本中创建 which 命令,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/28642253/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com