今天在写一个Shell脚本的时候,发现调试的时候会报一个错
[oracle@smidb11 autoawrrpt]$ . a.sh
dirname:无效选项 -- b
请尝试执行"dirname --help"来获取更多信息。
basename:无效选项 -- b
请尝试执行"basename --help"来获取更多信息。
脚本内容如下:
[oracle@smidb11 autoawrrpt]$ cat a.sh
#!/bin/bash
################################################################################
#自动生成最新的AWR报表
#版本:v1.0
#变量说明:
#LOGFILE:保存日志文件名
####################################修改历史####################################
#
################################################################################
WORKDIR=$(dirname $0) #获取到脚本所在目录
cd ${WORKDIR} #改变当前目录为脚本所在目录
WORKDIR=`pwd` #获取当前目录的具体路径
SCRIPTNAME=$(basename $0) #获取脚本名称
FPATH="${WORKDIR}\${SCRIPTNAME}" #脚本全路径
LOGFILE="${WORKDIR}/${SCRIPTNAME}.log" #设置日志文件名称
AWR_TYPE="html" #设置AWR的类型,值应该为html或者txt
AWR_HOME="${WORKDIR}/awr_report" #设置存放AWR报表的目录
AWR_DAYS=1 #设置AWR Snap检索的天数
HOSTNAME=`hostname` #主机名
mkdir -p ${AWR_HOME} #如果没有此目录,则创建 -p选项代表无则创建,有则直接返回
报错的是在 WORKDIR=$(dirname $0) 这一行内容,这让我感觉很奇怪,第一,我没有在这个命令后面添加-b选项啊,为什么会报错错误的选项 -b 呢?
然后在这行下面增加 echo $0 输出,增加后代码如下:
[oracle@smidb11 autoawrrpt]$ cat a.sh
#!/bin/bash
################################################################################
#自动生成最新的AWR报表
#版本:v1.0
#变量说明:
#LOGFILE:保存日志文件名
####################################修改历史####################################
#
################################################################################
WORKDIR=$(dirname $0) #获取到脚本所在目录
cd ${WORKDIR} #改变当前目录为脚本所在目录
WORKDIR=`pwd` #获取当前目录的具体路径
echo $0
SCRIPTNAME=$(basename $0) #获取脚本名称
FPATH="${WORKDIR}\${SCRIPTNAME}" #脚本全路径
LOGFILE="${WORKDIR}/${SCRIPTNAME}.log" #设置日志文件名称
AWR_TYPE="html" #设置AWR的类型,值应该为html或者txt
AWR_HOME="${WORKDIR}/awr_report" #设置存放AWR报表的目录
AWR_DAYS=1 #设置AWR Snap检索的天数
HOSTNAME=`hostname` #主机名
mkdir -p ${AWR_HOME} #如果没有此目录,则创建 -p选项代表无则创建,有则直接返回
看看$0的内容到底是什么,运行脚本结果如下:
[oracle@smidb11 autoawrrpt]$ . a.sh
dirname:无效选项 -- b
请尝试执行"dirname --help"来获取更多信息。
-bash
basename:无效选项 -- b
请尝试执行"basename --help"来获取更多信息。
这就奇怪了,$0 不是应该返回脚本的当前路径吗? 怎么会变成 -base 了呢?
我们接着试验:
[oracle@smidb11 autoawrrpt]$ ./a.sh
./a.sh
直接运行,输出结果变成正确的啦,$0输出执行脚本的相对路径.
为什么会导致这样的问题呢?为了证明这个问题,我们直接在shell里面运行 echo $0
[mysql@test ~]$ echo $0
-bash
然后我们回顾一下,Linux下面的点命令
使用 点命令执行脚本的时候,直接在当前的shell中执行脚本,而不会打开一个新的Shell执行脚本
看到这里我们就不难理解了。因为 . 命令不会打开新的Shell执行脚本,所以echo $0的时候,$0的值还是我们原来的shell的$0, 也就是 -bash 了, 而 basename命令就会把 -bash当成一个参数来执行,所以才会报错,错误的选项 -b。
知道原因了,我们的问题也就很好解决了,直接运行,不要通过 点命令 或者 source 命令运行脚本即可。