如果我有一个文件路径,例如…
/home/smith/Desktop/Test
/home/smith/Desktop/Test/
我如何改变字符串,使它将是父目录?
e.g.
/home/smith/Desktop
/home/smith/Desktop/
如果我有一个文件路径,例如…
/home/smith/Desktop/Test
/home/smith/Desktop/Test/
我如何改变字符串,使它将是父目录?
e.g.
/home/smith/Desktop
/home/smith/Desktop/
当前回答
dir=/home/smith/Desktop/Test
parentdir="$(dirname "$dir")"
如果后面有斜杠,也可以。
其他回答
只需使用echo $(cd ../ && pwd),同时在您想要查找的父目录中工作。这个链还有一个额外的好处,就是没有后面的斜杠。
另一个答案的动机
我喜欢非常简短、清晰、有保证的代码。如果它不运行外部程序,它的好处是,因为有一天你需要处理大量的输入,它会明显更快。
原则
不确定你有什么保证,想要什么,所以还是提供吧。
如果你有保证,你可以用很短的代码完成它。其思想是使用bash文本替换特性来删除最后一个斜杠和后面的任何内容。
回答从简单到较复杂情况下的原题。
如果路径保证结尾没有任何斜杠(in和out)
P=/home/smith/Desktop/Test ; echo "${P%/*}"
/home/smith/Desktop
如果path保证以一个斜杠结尾(in和out)
P=/home/smith/Desktop/Test/ ; echo "${P%/*/}/"
/home/smith/Desktop/
如果输入路径可能以零或一个斜杠(而不是多个)结束,而您希望输出路径以不带斜杠结束
for P in \
/home/smith/Desktop/Test \
/home/smith/Desktop/Test/
do
P_ENDNOSLASH="${P%/}" ; echo "${P_ENDNOSLASH%/*}"
done
/home/smith/Desktop
/home/smith/Desktop
如果输入路径可能有许多多余的斜杠,而您希望输出路径结束时不带斜杠
for P in \
/home/smith/Desktop/Test \
/home/smith/Desktop/Test/ \
/home/smith///Desktop////Test//
do
P_NODUPSLASH="${P//\/*(\/)/\/}"
P_ENDNOSLASH="${P_NODUPSLASH%%/}"
echo "${P_ENDNOSLASH%/*}";
done
/home/smith/Desktop
/home/smith/Desktop
/home/smith/Desktop
显然,父目录是通过简单地附加点。点文件名来给出的:
/home/smith/Desktop/Test/.. # unresolved path
但你必须要解析路径(没有任何点-点路径组件的绝对路径):
/home/smith/Desktop # resolved path
使用dirname的顶部答案的问题是,当你输入一个带有点号的路径时,它们不起作用:
$ dir=~/Library/../Desktop/../..
$ parentdir="$(dirname "$dir")"
$ echo $parentdir
/Users/username/Library/../Desktop/.. # not fully resolved
这个更强大:
dir=/home/smith/Desktop/Test
parentdir=$(builtin cd $dir; pwd)
你可以在/home/smith/Desktop/Test/..,但也有更复杂的路径,比如:
$ dir=~/Library/../Desktop/../..
$ parentdir=$(builtin cd $dir; pwd)
$ echo $parentdir
/Users # the fully resolved path!
注意:使用内置确保不会调用用户定义的cd函数变体,而是调用没有输出的默认实用程序形式。
如果/home/smith/Desktop/Test/../是你想要的:
dirname 'path/to/child/dir'
如图所示。
dir=/home/smith/Desktop/Test
parentdir="$(dirname "$dir")"
如果后面有斜杠,也可以。