@(Linux 命令脚本)
编写脚本中,需要获取执行脚本的绝对路径,今天写脚本的时候不小心踩了个坑,记录下。
那个坑的脚本大概是这样的:
#!/bin/bash
work_path=$(dirname $0)
cd ~/${work_path}
## blblbl
cd /home/lcd/
cp ./something ${work_path}
问题就出在最后那句,本意我是希望把/home/lcd/something 复制到我脚本的执行目录。
假设我的脚本目录在/home/lcd/shell/下,
bash /home/lcd/shell/mytest.sh
# work_path -> home/lcd/shell
# 能得到想要的
lcd:/home/lcd/shell/ $ bash ./mytest.sh
# work_path -> ./
# 所以上面的执行方式,something 还是复制到了./
# 也就是最后 cd 进去的目录下
查了下,总结下获取路径的方法,以及可能的误区。
- 脚本参数 $0
使用 $0 可以获取到路径,但不一定是绝对路径,实际上, $0 是代表传递给 bash 这些的第一个参数。
$ bash ./mytest.sh # $0= ./mytest.sh
$ bash ./shell/mytest.sh # $0= ./shell/mytest.sh
$ bash /home/lcd/shell/mytest.sh # $0= /home/lcd/shell/mytest.sh
如果执行的时候给的是绝对路径,那么我们可以通过 $0 提取到绝对路径,实际上我们没法保证。
- 使用 pwd 获取路径
pwd 可以打印当前路径,但是也不一定是脚本的位置。
lcd:/home/lcd/shell/ $ bash ./mytest.sh
# pwd=/home/lcd/shell/
lcd:/home/lcd/ $ bash ./shell/mytest.sh
# pwd=/home/lcd/
lcd:/root/ $bash /home/lcd/shell/mytest.sh
# pwd=/root/
可见,只有第一条的执行情况才满足实际需要,pwd 打印的是你站在那里,而不是脚本在哪里。
从上面两点,看看下面这些获取路径的方法
# 在某些情况下会拿到错误结果
work_path=$(dirname $0)
work_path=$(pwd)
## 正确实现
# 通过 readlink 获取绝对路径,再取出目录
work_path=$(dirname $(readlink -f $0))
# 或者曲线救国
work_path=$(dirname $0)
cd ./${work_path} # 当前位置跳到脚本位置
work_path=$(pwd) # 取到脚本目录