我正在寻找一个可以在 Bash 中使用的简单命令,用于在 OS X 上查找文件的绝对和规范路径(类似于 Linux 下的“readlink -f”)。
以下示例 bash 会话描述了一个名为“abspath”的[虚构]实用程序,它表现出所需的行为:
$ pwd
/Users/guyfleegman
$ ls -lR
drwxr-xr-x 4 guyfleegman crew 136 Oct 30 02:09 foo
./foo:
-rw-r--r-- 1 guyfleegman crew 0 Oct 30 02:07 bar.txt
lrwxr-xr-x 1 guyfleegman crew 7 Oct 30 02:09 baz.txt -> bar.txt
$ abspath .
/Users/guyfleegman
$ abspath foo
/Users/guyfleegman/foo
$ abspath ./foo/bar.txt
/Users/guyfleegman/foo/bar.txt
$ abspath foo/baz.txt
/Users/guyfleegman/foo/baz.txt
与上例中最后一次调用“abspath”一样,我希望它不会自动解析符号链接,但在这里我不会太挑剔。
答案1
function abspath() { pushd . > /dev/null; if [ -d "$1" ]; then cd "$1"; dirs -l +0; else cd "`dirname \"$1\"`"; cur_dir=`dirs -l +0`; if [ "$cur_dir" == "/" ]; then echo "$cur_dir`basename \"$1\"`"; else echo "$cur_dir/`basename \"$1\"`"; fi; fi; popd > /dev/null; }
例子:
abspath / => /
abspath /.DS_Store => /.DS_Store
abspath ~ => /Users/mschrag
cd /tmp; abspath . => /tmp
cd /; abspath .DS_Store => /.DS_Store
答案2
我认为没有内置命令可以做到这一点。杰西·威尔逊为此编写了一个 bash 脚本:
#!/bin/bash
cd -P -- "$(dirname -- "$1")" &&
printf '%s\n' "$(pwd -P)/$(basename -- "$1")"
但是,它对于 下面的路径效果不佳/
,例如/etc
(打印//etc
),以及 和.
(两种情况下都..
打印/cwd/.
)。我尝试修改它,但我的 bash-fu 不够好,让我失望了。
这是我的建议:
#!/usr/bin/env python
import os.path
import sys
for arg in sys.argv[1:]:
print os.path.abspath(arg)
另存为/usr/bin/abspath
或类似名称并使其可执行。示例输出:
Servus08:~ danielbeck$ abspath .
/Users/danielbeck
Servus08:~ danielbeck$ abspath /tmp
/tmp
Servus08:~ danielbeck$ abspath Documents
/Users/danielbeck/Documents
Servus08:~ danielbeck$ abspath . /tmp Documents
/Users/danielbeck
/tmp
/Users/danielbeck/Documents
如果你做想要符号链接解析,请将print
以下行更改如下:
print os.path.realpath(os.path.abspath(arg))
得到这个:
Servus08:~ danielbeck$ abspath . /tmp Documents
/Users/danielbeck
/private/tmp
/Users/danielbeck/Documents
答案3
一种选择是只安装 coreutils 并使用greadlink -f
。它可以解析符号链接,并且如果不存在,它可以与/Foo/
或一起使用,但不能与if不存在一起使用。~/foo.txt
/Foo/foo.txt
/Foo/
$ brew install coreutils
$ greadlink -f /etc
/private/etc
$ greadlink -f ~/Documents/
/Users/lauri/Documents
$ greadlink -f ..
/Users
$ greadlink -f //etc/..////
/private
$ greadlink -f /Foo
/Foo
$ greadlink -f /Foo/foo.txt
$
这不能解决符号链接问题,也不能与/Foo/foo.txt
任何一种一起工作。
abspath() {
if [ -d "$1" ]; then
( cd "$1"; dirs -l +0 )
else
( cd "$(dirname "$1")"; d=$(dirs -l +0); echo "${d%/}/${1##*/}" )
fi
}
abspath /etc # /etc
abspath ~/Foo/foo.txt # doesn't work
abspath ~/Foo # works
abspath .
abspath ./
abspath ../
abspath ..
abspath /
abspath ~
abspath ~/
abspath ~/Documents
abspath /\"\ \'
abspath /etc/../etc/
abspath /private//etc/
abspath /private//
abspath //private # //private
abspath ./aa.txt
abspath aa.tar.gz
abspath .aa.txt
abspath /.DS_Store
abspath ~/Documents/Books/
dirs -l
执行波浪号扩展。dirs +0
如果堆栈中还有其他目录,则仅打印最顶层目录。
答案4
如果你已经为 perl 安装了 File::Spec 模块,你可以这样做:
perl -MFile::Spec -e 'print File::Spec->rel2abs("../however/complex/../you/want/to.get"), "\n"'