[转]linux shell 获取当前正在执行脚本的绝对路径

本文详细解析了如何正确获取当前运行脚本的路径,纠正了使用`pwd`命令和`$0`变量的误区,并提供了一个实用的解决方案。通过使用`dirname`、`cd`和`pwd`命令组合,可以准确地获得脚本所在的基路径。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

原文链接:http://sexywp.com/bash-how-to-get-the-basepath-of-current-running-script.htm

 

常见的一种误区,是使用 pwd 命令,该命令的作用是“print name of current/working directory”,这才是此命令的真实含义,当前的工作目录,这里没有任何意思说明,这个目录就是脚本存放的目录。所以,这是不对的。你可以试试 bash shell/a.sh,a.sh 内容是 pwd,你会发现,显示的是执行命令的路径 /home/june,并不是 a.sh 所在路径:/home/june/shell/a.sh

 

  另一个误人子弟的答案,是 $0,这个也是不对的,这个$0是Bash环境下的特殊变量,其真实含义是:

   Expands to the name of the shell or shell script. This is set at shell initialization.  If bash is invoked with a file of commands, $0 is set to the name of that file. If bash is started with the -c option, then $0 is set to the first argument after the string to be executed, if one is present. Otherwise, it is set to the file name used to invoke bash, as given by argument zero.

   这个$0有可能是好几种值,跟调用的方式有关系:

  • 使用一个文件调用bash,那$0的值,是那个文件的名字(没说是绝对路径噢)

  • 使用-c选项启动bash的话,真正执行的命令会从一个字符串中读取,字符串后面如果还有别的参数的话,使用从$0开始的特殊变量引用(跟路径无关了)

  • 除此以外,$0会被设置成调用bash的那个文件的名字(没说是绝对路径)

下面对比下正确答案:

 

 

basepath=$(cd `dirname $0`; pwd)

 

在此解释下basepath :

dirname $0,取得当前执行的脚本文件的父目录

cd `dirname $0`,进入这个目录(切换当前工作目录)

pwd,显示当前工作目录(cd执行后的)

由此,我们获得了当前正在执行的脚本的存放路径。

转载于:https://www.cnblogs.com/FlyFive/p/3640267.html

<think>好的,用户想了解在Shell脚本中如何获取当前运行脚本绝对路径。我需要先回忆相关的知识点,确保信息的准确性。首先,Shell有不同的方式获取脚本路径,但需要注意不同方法之间的差异以及兼容性问题。 我记得常用的方法是使用Bash的`$BASH_SOURCE`变量。这个变量在Bash中指向当前执行脚本路径。不过,用户可能使用的是不同的Shell,比如sh,这时候可能需要用`$0`。但要注意,当脚本被source时,`$0`可能会返回shell的名称而不是脚本路径,所以需要处理这种情况。 然后,结合dirname和realpath或者readlink来获取绝对路径。例如,`dirname $BASH_SOURCE`可以得到脚本的目录,但可能还是相对路径。所以需要用realpath或者readlink换为绝对路径。不过,不同的系统可能这些命令的参数不一样,比如在macOS上,realpath可能默认不可用,需要安装coreutils,或者改用其他方法。 另外,考虑到跨平台的兼容性,可能需要使用更通用的方法。比如,使用`cd`和`pwd`的组合来获取绝对路径。例如:`SCRIPT_DIR=$(cd -- "$(dirname -- "$BASH_SOURCE")" &>/dev/null && pwd)`。这样可以处理符号链接的情况吗?或者是否需要添加`-P`参数来解析物理路径? 还有,当脚本被符号链接时,如何获取实际的文件路径?这时候可能需要使用`readlink -f`,但在macOS上,`readlink`可能不支持`-f`参数,需要改用`greadlink`。但这样又会增加复杂性,用户可能需要判断系统类型。 需要提醒用户注意不同Shell和系统的差异,尤其是Bash和sh之间的区别。比如,在sh中,`$BASH_SOURCE`不可用,只能用`$0`。但`$0`在source脚本时会变成shell的路径,所以需要检查是否被source执行的情况。 可能还需要给出几个不同的示例,说明在不同情况下如何正确获取路径,并解释每个步骤的作用。同时,指出潜在的陷阱,比如空格在路径中导致的问题,需要用引号包裹变量。 总结一下,回答的结构应该是先介绍基本方法,然后讨论兼容性和不同情况下的处理,最后给出推荐的可靠方法,并说明注意事项。这样用户可以根据自己的需求选择合适的方式,并避免常见错误。</think>在Shell脚本获取当前运行脚本绝对路径,推荐使用以下两种最可靠的方式: 一、通用兼容方案(推荐) ```bash #!/bin/bash SCRIPT_DIR=$(cd -- "$(dirname -- "${BASH_SOURCE[0]}")" &>/dev/null && pwd) SCRIPT_PATH="${SCRIPT_DIR}/$(basename "${BASH_SOURCE[0]}")" echo "脚本目录:${SCRIPT_DIR}" echo "完整路径:${SCRIPT_PATH}" ``` 二、支持realpath命令的方案(更简洁) ```bash #!/bin/bash SCRIPT_PATH=$(realpath "${BASH_SOURCE[0]}") SCRIPT_DIR=$(dirname "${SCRIPT_PATH}") echo "脚本目录:${SCRIPT_DIR}" echo "完整路径:${SCRIPT_PATH}" ``` 注意事项: 1. 必须使用`${BASH_SOURCE[0]}`而不是`$0`: - `$0`在脚本被source执行时会失效(显示为shell名称) - `${BASH_SOURCE[0]}`始终指向当前脚本路径 2. 路径中的特殊字符处理: - 所有路径变量必须用双引号包裹(处理空格等特殊字符) - 使用`cd --`确保以`-`开头的路径被正确处理 3. 不同系统差异: - macOS需要先安装coreutils才能使用`realpath`命令 ```bash brew install coreutils # 安装后使用grealpath ``` 4. 符号链接处理: - `realpath`默认会解析符号链接 - 若需要保留符号链接,使用: ```bash realpath -s "${BASH_SOURCE[0]}" # 不解析符号链接 ``` 5. 安全判断逻辑(增强健壮性): ```bash if [[ "$(basename "${SCRIPT_PATH}")" == "$(basename "$0")" ]]; then echo "脚本直接执行" else echo "脚本被source执行" fi ``` 典型应用场景示例: ```bash #!/bin/bash # 获取脚本所在目录的绝对路径 script_dir=$(cd -- "$(dirname -- "${BASH_SOURCE[0]}")" &>/dev/null && pwd) # 加载同目录下的配置文件 source "${script_dir}/config.cfg" # 切换到脚本所在目录执行操作 (cd "${script_dir}" && ./sub_script.sh) # 访问同级资源文件 icon_path="${script_dir}/assets/icon.png" ``` 各方案对比: | 方法 | 优点 | 缺点 | 适用场景 | |-------------------|-----------------------|-------------------------|---------------------| | cd + pwd组合 | 无需额外命令 | 需要处理路径拼接 | 通用环境 | | realpath方案 | 语法简洁 | 依赖额外软件包 | Linux系统 | | readlink -f方案 | 直接解析符号链接 | macOS不原生支持 | 需要解析符号链接 | 扩展知识: 当脚本被不同方式调用时路径的表现: ``` 直接执行: ./test.sh → $0 = ./test.sh bash执行bash test.sh → $0 = test.sh source执行: source test.sh → $0 = -bashBASH_SOURCE仍有效) 符号链接: ln -s test.sh link.sh → 可解析真实路径 ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值