且构网

分享程序员开发的那些事...
且构网 - 分享程序员编程开发的那些事

如何从脚本本身中获取Bash脚本的源目录?

更新时间:2023-12-05 14:22:28

#!/bin/bash

DIR="$( cd "$( dirname "${BASH_SOURCE[0]}" )" &> /dev/null && pwd )"

是有用的单行代码,无论从何处调用脚本,它都会为您提供脚本的完整目录名称.

is a useful one-liner which will give you the full directory name of the script no matter where it is being called from.

只要用于查找脚本的路径的最后一个组成部分不是符号链接(目录链接都可以),它将起作用.如果您还想解析到脚本本身的任何链接,则需要一个多行解决方案:

It will work as long as the last component of the path used to find the script is not a symlink (directory links are OK). If you also want to resolve any links to the script itself, you need a multi-line solution:

#!/bin/bash

SOURCE="${BASH_SOURCE[0]}"
while [ -h "$SOURCE" ]; do # resolve $SOURCE until the file is no longer a symlink
  DIR="$( cd -P "$( dirname "$SOURCE" )" >/dev/null 2>&1 && pwd )"
  SOURCE="$(readlink "$SOURCE")"
  [[ $SOURCE != /* ]] && SOURCE="$DIR/$SOURCE" # if $SOURCE was a relative symlink, we need to resolve it relative to the path where the symlink file was located
done
DIR="$( cd -P "$( dirname "$SOURCE" )" >/dev/null 2>&1 && pwd )"

最后一个将与别名, source bash -c ,符号链接等的任意组合一起使用.

This last one will work with any combination of aliases, source, bash -c, symlinks, etc.

请注意:如果在运行此代码段之前 cd 到另一个目录,则结果可能不正确!

Beware: if you cd to a different directory before running this snippet, the result may be incorrect!

此外,请注意 $ CDPATH 陷阱,如果用户巧妙地覆盖了cd以将输出重定向到stderr(包括转义序列,例如调用 update_terminal_cwd& 2 (在Mac上).在 cd 命令的末尾添加>/dev/null 2>& 1 会同时考虑到这两种可能性.

Also, watch out for $CDPATH gotchas, and stderr output side effects if the user has smartly overridden cd to redirect output to stderr instead (including escape sequences, such as when calling update_terminal_cwd >&2 on Mac). Adding >/dev/null 2>&1 at the end of your cd command will take care of both possibilities.

要了解其工作原理,请尝试运行以下更详细的形式:

To understand how it works, try running this more verbose form:

#!/bin/bash

SOURCE="${BASH_SOURCE[0]}"
while [ -h "$SOURCE" ]; do # resolve $SOURCE until the file is no longer a symlink
  TARGET="$(readlink "$SOURCE")"
  if [[ $TARGET == /* ]]; then
    echo "SOURCE '$SOURCE' is an absolute symlink to '$TARGET'"
    SOURCE="$TARGET"
  else
    DIR="$( dirname "$SOURCE" )"
    echo "SOURCE '$SOURCE' is a relative symlink to '$TARGET' (relative to '$DIR')"
    SOURCE="$DIR/$TARGET" # if $SOURCE was a relative symlink, we need to resolve it relative to the path where the symlink file was located
  fi
done
echo "SOURCE is '$SOURCE'"
RDIR="$( dirname "$SOURCE" )"
DIR="$( cd -P "$( dirname "$SOURCE" )" >/dev/null 2>&1 && pwd )"
if [ "$DIR" != "$RDIR" ]; then
  echo "DIR '$RDIR' resolves to '$DIR'"
fi
echo "DIR is '$DIR'"

它将显示类似以下内容的

And it will print something like:

SOURCE './scriptdir.sh' is a relative symlink to 'sym2/scriptdir.sh' (relative to '.')
SOURCE is './sym2/scriptdir.sh'
DIR './sym2' resolves to '/home/ubuntu/dotfiles/fo fo/real/real1/real2'
DIR is '/home/ubuntu/dotfiles/fo fo/real/real1/real2'