共计 4360 个字符,预计需要花费 11 分钟才能阅读完成。
在 Linux 中,当须要 cd 到多层的下级目录时,须要输出 cd ../../../
等等多个“../”。
在输出比拟快的时候,往往两头会多输出一个点号‘.’、或者少输出一个点号‘.’,须要往前删除,从新输出。
而在理论开发工作中,特地是进行 Android 零碎开发工作时,源代码目录可能会有多级子目录。
当进入到目录构造十分深的子目录时,想要返回到特定的多层下级目录,须要手动输出多个“../”,十分麻烦,也很容易输错。
为了简化输出,缩小输出出错的状况,本篇文章介绍一个名为 cdup.sh
的 shell 脚本。
该脚本接管一个整数参数,指定要返回到几层的下级目录,能够十分不便地返回到多层下级目录,进步工作效率。
例如,执行 source cdup.sh 5
命令,等价于执行 cd ../../../../../
命令。
所给的数字 5 指定要返回到第 5 层下级目录。
前面会介绍如何设置命令别名来防止输出 source cdup.sh
这些字符,能够简化成 up 5
这样的输出即可。
因为 shell 脚本默认运行在子 shell 外面,而 cd
命令只能扭转以后 shell 的工作目录。
所以在 shell 脚本外面执行 cd
命令不能扭转父 shell 的工作目录。
为了让脚本执行完结后,还放弃在 cd
后的目录,须要用 source
命令来执行 shell 脚本。
应用 source
命令执行 shell 脚本,会运行在以后 shell 下,而不是运行在子 shell 外面。
脚本代码
列出 cdup.sh
脚本的具体代码如下所示。
在这个代码中,对大部分要害代码都提供了具体的正文,不便浏览。
这篇文章的前面也会对一些关键点进行阐明,有助了解。
#!/bin/bash
# 当要 cd 到多层的下级目录时, 须要输出 cd ../../../ 等等多个 "../".
# 为了简化输出, 以后脚本能够解决一个整数参数, 指定返回到几层的下级目录.
# 例如 source cdup.sh 3 等价于 cd ../../../
# 为了让脚本执行完结后, 还放弃在 cd 后的目录, 须要用 source 命令
# 来执行该脚本. 能够在 ~/.bashrc 文件中增加如下别名来不便执行:
# alias up='source cdup.sh'
# 后续执行 up 3 命令, 就等价于 cd ../../../
# 这里假如 cdup.sh 脚本放在 PATH 指定的寻址目录下. 例如 /usr/bin 目录.
# 如果 cdup.sh 脚本没有放在默认的寻址目录下, 请指定残缺的绝对路径.
cdup_show_help()
{
printf "USAGE
source cdup.sh number
OPTIONS
number: 指定要返回到几层的下级目录.
例如 source cdup.sh 3 等价于 cd ../../../
NOTE
能够应用 alias up='source cdup.sh' 设置 up 别名来不便执行.
"
}
if [$# -ne 1]; then
cdup_show_help
# 以后脚本预期通过 source 命令来调用, 不能执行 exit 命令,
# 否则会退出调用该脚本的 shell. 上面通过 return 命令来返回.
return 1
fi
UPDIR_PATH="../"
# 依据传入的数字参数, 计算要返回到几层下级目录, 并将后果写到规范输入
count_updirs()
{
# 所给的第一个参数指定要返回到几层下级目录
local count=$1
local updirs=""
while ((--count >= 0)); do
# 没有应用算术扩大时, bash 的 += 运算符默认会拼接字符串.
# 上面语句会拼接多个 "../", 失去相似于 "../../../" 的成果.
updirs+=${UPDIR_PATH}
done
echo ${updirs}
}
target_dir="$(count_updirs $1)"
# 应用 \cd 来指定不应用 alias 别名, 执行原始的 cd 命令.
\cd "${target_dir}"
return
代码关键点阐明
倡议设置命令别名来执行以后脚本
如后面阐明,须要应用 source
命令来执行 cdup.sh
脚本,以便执行该脚本之后,能够放弃在 cd
后的目录下。
即,执行的时候,须要写为 source cdup.sh
。
这样须要输出比拟多的字符,而且也容易遗记提供 source
命令。
为了不便输出,在脚本正文中,倡议设置命令别名来执行以后脚本。
例如,在 ~/.bashrc
文件中增加上面语句来设置命令别名:
alias up='source cdup.sh'
增加这个语句后,在以后终端中,须要执行 source ~/.bashrc
命令,这个别名才会失效。
也能够从新关上终端,在新关上的终端中,这个别名默认就会失效。
在别名失效之后,就能够应用 up 命令来执行 cdup.sh
脚本。
例如,up 3
命令等价于 source cdup.sh 3
命令。
这里假如 cdup.sh
脚本放在 PATH 全局变量指定的寻址目录外面,通过文件名就能够执行,不须要指定文件的门路。
如果该脚本没有放在默认的寻址目录外面,须要提供文件的绝对路径。
在 cd 命令后面加反斜线指定不应用别名
在 bash 中,反对设置命令别名。
如果某个命令别名被设置成 cd 字符串,那么执行 cd 命令,会执行该别名指定的命令。
即,不会执行本来切换工作目录的 cd
命令。
为了防止这个问题,能够应用 \cd
来指定不应用别名,会执行 cd
命令本身。
在 cdup.sh
脚本中应用了 \cd
这个写法,以确保能够切换工作目录。
查看 man bash 对 alias 别名的阐明,没有明确提到在命令后面加反斜线能够不应用别名。
这是基于反斜线转义字符的作用、以及别名的解决逻辑所引申进去的非凡用法。
对 \cd
这种写法能够不应用命令别名的相干阐明具体解读如下。
查看 man bash 对反斜线 \
的阐明如下:
A non-quoted backslash (\) is the escape character.
It preserves the literal value of the next character that follows, with the exception of <newline>.
即,当反斜线 \
没有被任何引号括起来时,它能够放弃跟在前面的下一个字符为本身不变。
且,通过 bash 解决之后,会去掉 \
字符,只保留下一个字符本身。
例如,在 bash 中,没有加任何引号的状况下,\c
会失去字符 c。
具体举例说明如下:
$ echo \c
c
$ echo \cd
cd
能够看到,echo \c
命令打印的是字符 c,而不是 “c” 字符串。
echo \cd
命令打印的是 “cd” 字符串。
实际上,这是 \c
失去字符 c 之后,字符 c 再跟字符 d 组合成 “cd” 字符串。
查看 man bash 对 alias 别名的阐明如下:
The first word of each simple command, if unquoted, is checked to see if it has an alias.
If so, that word is replaced by the text of the alias.
即,bash 会获取命令第一个没有被引号括起来的字符,基于这个字符来查看是不是一个别名。
如果是,就把这个命令别名替换成指定的命令。
在 bash 中,无奈把命令别名的第一个字符设成反斜线。
具体举例如下:
$ alias \testcd='cd ../../'
$ alias | grep cd
alias testcd='cd ../../'
这里先执行 alias \testcd='cd ../../'
命令设置别名。
从输出的参数来看,所设置的别名看起来是 \testcd
。
然而用 alias
命令打印出所有命令别名,并用 grep
过滤出蕴含 cd
的行。
能够看到,理论设置的别名是 testcd。
这个别名的第一个字符并不是反斜线。
如后面阐明,bash 会去掉没有用引号括起来的反斜线。
查看 GNU bash 的在线帮忙链接 https://www.gnu.org/software/…,有如下阐明:
3.1.1 Shell Operation
2.Breaks the input into words and operators, obeying the quoting rules described in Quoting. These tokens are separated by metacharacters. Alias expansion is performed by this step.
4.Performs the various shell expansions, breaking the expanded tokens into lists of filenames and commands and arguments.
3.5.9 Quote Removal
After the preceding expansions, all unquoted occurrences of the characters‘’,‘’’, and‘”’that did not result from one of the above expansions are removed.
即,bash 在 Quote Removal 阶段会移除反斜线 \
字符。
而 Quote Removal 是在 shell 扩大之后进行。
判断别名是在 shell 扩大之前进行。
所以,bash 在判断别名时,还会看到反斜线 \
字符。
也就是说,执行 \cd
命令时,在判断别名阶段,会看到这个命令的第一个字符是 \
。
如后面阐明,无奈把命令别名的第一个字符设成反斜线。
那么找不到 \cd
对应的别名,不会进行别名替换。
通过 Quote Removal 阶段后,去掉了反斜线,只剩下 cd
命令。
此时曾经过了判断别名阶段,所以执行的是 cd
命令本身,不会执行名为 cd 的命令别名。
即,严格来说,在命令后面加反斜线,并不是不应用别名。
而是会找不到以反斜线结尾的别名,所以没有进行别名扩大。
艰深来说,能够简略了解为,在命令后面加反斜线指定不应用别名。
执行以后脚本的例子
应用后面阐明的 up
命令别名来执行 source cdup.sh
命令,具体执行后果如下:
[frameworks/base/services/core/java/com/android/server]$ cd ../../../../../
[frameworks/base/services]$ cd -
frameworks/base/services/core/java/com/android/server
[frameworks/base/services/core/java/com/android/server]$ up 5
[frameworks/base/services]$
在这个例子中,方括号两头显示的目录门路就是以后 shell 的工作目录。
能够看到,cd ../../../../../
命令退回到第 5 层下级目录。
而后执行 cd -
命令返回到原来的子目录下。
再执行 up 5
命令,也是退回到第 5 层下级目录。
up 5
命令的成果跟 cd ../../../../../
命令雷同,然而输出非常简单,也不容易出错。