Linux 情景 linux--shell 脚本中的路径问题

stack · 2017年04月21日 · 1699 次阅读

情景

使用 shell 编写脚本时,经常要使用到目录和文件。而有时候,对于目录和文件的使用总是达不到理想效果——目录和文件总是不能在预想的位置创建,又或者,当把脚本移动到别的目录执行时,却不得不对修改脚本中文件或目录的路径,否则便不能使用,等等。

举个简单的例子来说明这种情景。
~/temp目录下创建两个目录:test1test2

$ pwd
/home/j-tester/temp
$ ls -1
test1
test2

test1目录下创建一个名称为getDate.sh的 shell 脚本,内容如下:

$ pwd
/home/j-tester/temp/test1
$ cat getDate.sh 
date +%Y%m%d-%H%M%S > shijian.txt

脚本的需求是:
执行date +%Y%m%d-%H%M%S命令,并将命令的结果保存到该脚本所在的目录下的shijian.txt文件中。

为其赋予执行权限,然后运行它:

$ chmod +x getDate.sh
$ ./getDate.sh

查看当前目录,确实产生了一个名为shijian.txt的文件,内容是执行脚本时的时间20170421-121406

$ ls -1
getDate.sh
shijian.txt
$ cat shijian.txt
20170421-121406

如果我们切换目录到../test2/,然后执行getDate.sh会发生什么呢?

$ cd ../test2/
$ pwd
/home/j-tester/temp/test2

切换了目录,执行时可以用绝对路径执行,也可以用相对路径来执行,这里使用了相对路径:

$ ../test1/getDate.sh

脚本执行后,查看../test1/目录下的shijian.txt文件,内容还是之前的时间20170421-121406,这说明脚本的执行并未对test1目录下的shijian.txt文件产生效果:

$ ls -1 ../test1/
getDate.sh
shijian.txt
$ cat ../test1/shijian.txt 
20170421-121406

而查看此时的当前目录(/home/j-tester/temp/test2),却有一个shijian.txt文件,内容中的时间比之前的时间要新20170421-121833,而这正是进入到test2目录之后执行脚本时的时间:

$ ls -1
shijian.txt
$ cat shijian.txt 
20170421-121833

由上,可以预见的是,在不同的目录下执行getDate.sh文件,都会在这些目录下产生shijian.txt文件。这与预期的在getDate.sh脚本所在的目录下产生shijian.txt文件的需求是不符的。

其实,不仅仅是上面这个脚本,许多脚本都有这样的现象:只有在进入了脚本所在的目录下执行脚本,才会产生预期的效果。在 windows 系列系统中也有软件有这样的情况:一个可执行文件,使用绝对路径或相对路径执行它时会没有效果或者干脆直接报错,而只有先进入到这个可执行文件所在的目录后,再执行它才可以正常使用。

这是因为,正常情况下,这类脚本预期的执行目录是它所在的目录,而当你在其它目录执行它时就会出现这样那样的问题。

本文主要讨论这类问题该如何解决。

方案

方案一

可以为用到的目录或文件设置绝对路径。

在上例中,可以用绝对路径表示shijian.txt,就可以实现无论在任何目录执行getDate.sh,其输出都是~/temp/test1/shijian.txt

$ cat ~/temp/test1/getDate.sh 
date +%Y%m%d-%H%M%S > ~/temp/test1/shijian.txt

这样做的弊端是:
getDate.sh移动到别的目录执行时,产生的文件shijian.txt还是原来的目录,而非新目录。所以,当移动脚本后,还需要修改脚本中的绝对路径。

方案二

既然脚本中使用的目录或文件期望基于脚本的所在目录,那么可以在脚本中先进入到这个目录里,后续要使用目录和文件时,都基于此目录,用相对路径描述。

具体做法:

我们知道,shell 中有几个特殊的变量,其中$0表示 shell 脚本本身的文件名。
命令dirname是去掉后接参数的非目录的后缀,所以dirname $0就是脚本所在的目录:

$ pwd
/home/j-tester/temp

修改脚本的内容如下:

$ cat ./test1/getDate.sh 
echo "shell name is:$0"
echo "shell dir is :`dirname $0`"
date +%Y%m%d-%H%M%S > shijian.txt

使用相对路径执行脚本:

$ ./test1/getDate.sh 
shell name is:./test1/getDate.sh
shell dir is :./test1

使用绝对路径执行脚本:

$ ~/temp/test1/getDate.sh 
shell name is:/home/j-tester/temp/test1/getDate.sh
shell dir is :/home/j-tester/temp/test1

在脚本所在目录,用相对路径执行脚本:

$ cd test1
$ ./getDate.sh 
shell name is:./getDate.sh
shell dir is :.

通过上述结果可知,我们可以使用命令替换的方式,先cd进入到脚本所在的目录,然后使用pwd命令返回目录的绝对路径,将这个路径保持到变量里,以便后续调用

更进一步,可以将获取脚本所在目录优化为一句话代码

path="$(cd "$(dirname $0)";pwd)"

如果要进入这个目录,只需执行:

cd ${path}

如果你还希望在脚本中使用你执行脚本时所在的目录,可以在这之前,把pwd命令的结果保存到一个变量里:

run_path=`pwd`

使用相对路径的好处在于:
无需因为脚本的绝对路径发生了变化而不得不修改脚本中的相关路径。

当然,有时候使用绝对路径是在所难免的,但只要你进行了这方面的处理,脚本就会变得更通用,维护成本低很多。

扩展知识

  • 相对路径和绝对路径
  • 特殊变量,如 $0
  • 命令替换 $()
  • 双引号""和反引号``的使用
如果觉得我的文章对您有用,请随意打赏。您的支持将鼓励我继续创作!
暂无回复。
需要 登录 后方可回复, 如果你还没有账号请点击这里 注册