情景
使用shell编写脚本时,经常要使用到目录和文件。而有时候,对于目录和文件的使用总是达不到理想效果——目录和文件总是不能在预想的位置创建,又或者,当把脚本移动到别的目录执行时,却不得不对修改脚本中文件或目录的路径,否则便不能使用,等等。
举个简单的例子来说明这种情景。
在~/temp
目录下创建两个目录:test1
和test2
:
$ 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
- 命令替换$()
- 双引号
""
和反引号``的使用