内置命令

Shell 内置命令,就是由 Bash Shell 自身提供的命令,而不是文件系统中的可执行脚本文件。可使用type来确定一个命令是否是内置命令或可执行文件:

1
type [-tpa] 命令名
参数 说明
无参数 显示出命令名是外部指令还是 bash 内建指令
-t 会将命令名以底下这些字眼显示出他的意义:
file :表示为外部可执行文件;
alias :表示该指令为命令别名所设定的名称;
builtin :表示该指令为 bash 内建的指令功能;
-p 如果后面接的可执行文件为外部指令时,会显示出完整文件名
-a 会由 PATH 变量定义的路径中,将所有和命令名相关的指令都列出来,包含别名

通常来说,内置命令会比外部命令执行得更快,执行外部命令时不但会触发磁盘 I/O,还需要 fork 出一个单独的进程来执行,执行完成后再退出。而执行内置命令相当于调用当前 Shell 进程的一个函数, 还是在当前Shell环境进程内, 减少了上下文切换。

内置命令列表

命令 说明
: 扩展参数列表,执行重定向操作
. 读取并执行指定文件中的命令(在当前 shell 环境中)
alias 为指定命令定义一个别名
bg 将作业以后台模式运行
bind 将键盘序列绑定到一个 readline 函数或宏
break 退出 for、while、select 或 until 循环
builtin 执行指定的 shell 内建命令
caller 返回活动子函数调用的上下文
cd 将当前目录切换为指定的目录
command 执行指定的命令,无需进行通常的 shell 查找
compgen 为指定单词生成可能的补全匹配
complete 显示指定的单词是如何补全的
compopt 修改指定单词的补全选项
continue 继续执行 for、while、select 或 until 循环的下一次迭代
declare 声明一个变量或变量类型。
dirs 显示当前存储目录的列表
disown 从进程作业表中刪除指定的作业
echo 将指定字符串输出到 STDOUT
enable 启用或禁用指定的内建shell命令
eval 将指定的参数拼接成一个命令,然后执行该命令
exec 用指定命令替换 shell 进程
exit 强制 shell 以指定的退出状态码退出
export 设置子 shell 进程可用的变量
fc 从历史记录中选择命令列表
fg 将作业以前台模式运行
getopts 分析指定的位置参数
hash 查找并记住指定命令的全路径名
help 显示帮助文件
history 显示命令历史记录
jobs 列出活动作业
kill 向指定的进程 ID(PID) 发送一个系统信号
let 计算一个数学表达式中的每个参数
local 在函数中创建一个作用域受限的变量
logout 退出登录 shell
mapfile 从 STDIN 读取数据行,并将其加入索引数组
popd 从目录栈中删除记录
printf 使用格式化字符串显示文本
pushd 向目录栈添加一个目录
pwd 显示当前工作目录的路径名
read 从 STDIN 读取一行数据并将其赋给一个变量
readarray 从 STDIN 读取数据行并将其放入索引数组
readonly 从 STDIN 读取一行数据并将其赋给一个不可修改的变量
return 强制函数以某个值退出,这个值可以被调用脚本提取
set 设置并显示环境变量的值和 shell 属性
shift 将位置参数依次向下降一个位置
shopt 打开/关闭控制 shell 可选行为的变量值
source 读取并执行指定文件中的命令(在当前 shell 环境中)
suspend 暂停 Shell 的执行,直到收到一个 SIGCONT 信号
test 基于指定条件返回退出状态码 0 或 1
times 显示累计的用户和系统时间
trap 如果收到了指定的系统信号,执行指定的命令
type 显示指定的单词如果作为命令将会如何被解释
typeset 声明一个变量或变量类型。
ulimit 为系统用户设置指定的资源的上限
umask 为新建的文件和目录设置默认权限
unalias 刪除指定的别名
unset 刪除指定的环境变量或 shell 属性
wait 等待指定的进程完成,并返回退出状态码

alias设置别名

alisa用于给命令创建别名。好处就是可以将经常操作比较复杂的命令进行设置别名,通过别名的操作提高工作效率。若该命令且不带任何参数,则显示当前 Shell 进程中的所有别名列表。

语法 说明 备注
alias 别名=”命令” 为命令定义一个别名,比如alias ll="ls -l --color=auto" 这里使用单引号或双引号都可以
unalias 别名 删除指定的别名 临时删除当前Shell的别名, 如果想永久删除必须去配置文件中手动删除
unalias -a 删除当前Shell环境中所 临时删除当前Shell的别名, 如果想永久删除必须去配置文件中手动删除

echo输出字符串

echo用于在终端输出字符串,并在最后默认加上换行符。

语法 说明
echo 字符串 换行输出,默认
echo -n 字符串 不换行输出
echo -e 字符串 转义输出,会对转义符特殊处理,而不是当作一般文字输出

echo支持的转义符,-e参数里面会特殊处理的符号

转义符 说明
\a 发出警告声
\b 删除前一个字符
\c 最后不加上换行符号,用于强制清除echo的结尾换行输出
\f 换行但光标仍旧停留在原来的位置
\n 换行且光标移至行首;
\r 光标移至行首,但不换行
\t 插入tab
\v \f相同
\\ 插入\字符
\nnn 插入nnn(八进制)所代表的ASCII字符;

read读取控制台输入

read用于从标准输入中读取数据并赋值给变量。如果没有进行重定向,默认就是从终端控制台读取用户输入的数据;如果进行了重定向,那么可以从文件中读取数据。

1
read [-options] [var1 var2 ...]

options表示选项;var表示用来存储数据的变量,可以有一个,也可以有多个。optionsvar都是可选的,如果没有提供变量名,那么读取的数据将存放到环境变量 REPLY 变量中。$REPLY保存read最后一个读入命令的数据。options支持的参数如下:

选项 说明
-a array 把读取的数据赋值给数组 array,从下标 0 开始。
-d delimiter 用字符串delimiter指定读取结束的位置,而不是一个换行符(读取到的数据不包括delimiter)。
-e 在获取用户输入的时候,对功能键进行编码转换,不会直接显式功能键对应的字符。
-n num 读取 num 个字符,而不是整行字符。
-p prompt 显示提示信息,提示内容为prompt
-r 原样读取(Raw mode),不把反斜杠字符解释为转义字符。
-s 静默模式(Silent mode),不会在屏幕上显示输入的字符。
当输入密码和其它确认信息的时候,这是很有必要的。
-t seconds 设置超时时间,单位为秒。如果用户没有在指定时间内输入完成,
那么 read 将会返回一个非 0 的退出状态,表示读取失败。
-u fd 使用文件描述符 fd 作为输入源,而不是标准输入,类似于重定向。

exit退出

exit用于退出当前Shell环境进程结束运行,并且可以返回一个状态码。一般使用 $? 可以获取退出状态码。

语法 说明
exit 正确退出,默认返回状态码0,代表命令执行成功
exit 非0数 错误退出,数字建议的范围0~255, 一般代表命令执行失败

exit可以结束当前Shell进程,当Shell 进程执行出错退出时,可以返回不同的状态值代表不同的错误。比如执行一个脚本文件里面操作一个文件时,可以返回1 表示文件不存在,2 表示文件没有读取权限,3 表示文件类型不对。

declare设置变量

declare命令可用来声明变量并设置变量的属性,也可用来显示shell函数。若不加上任何参数,则会显示全部的shell变量与函数(与执行set指令的效果相同)。

设置变量的属性

1
declare [+/-][aArxif][变量名称=设置值]
选项 说明
+/- “-“可用来指定变量的属性,”+”则是取消变量所设的属性。
a array,设置为普通索引数组
A Array,设置为key-value关联数组
r readonly,将变量设置为只读, 也可以使用readonly
x exprot,设置变量成为全局变量,也可以使用export
i int,设置为整型变量。变量默认为字符串类型,可更改为整型declare -i sum=100+300+50
f function,设置为一个函数变量

实现关联数组变量

关联数组也称为”键值对(key-value)“数组,键(key)也即字符串形式的数组下标,值(value)也即元素值。

1
declare -A 关联数组变量名=([字符串key1]=值1 [字符串key2]=值2 ...)

declare也可以用于定义普通索引数组,-a 参数创建普通或索引数组,-A 创建关联数组:
declare -a 关联数组变量名=(值1 值2 ...)
declare -a 关联数组变量名=([0]=值1 [1]=值2 ...)

示例:

1
2
3
4
5
6
declare -A arrkv=([name]="abc" [age]=20 ["hobby"]="shell")
echo ${arrkv[name]} #abc
echo ${arrkv[age]} #20
echo ${arrkv[*]} #shell abc 20
echo ${arrkv[@]} #shell abc 20 与echo ${arrkv[*]}效果一样
echo ${#arrkv[@]} #3

注意:使用参数-A 才是关联数组 , 关联数组无法使用索引获取,不用declare -A实现的不是关联数组,而是只有最后一个值赋值成功的索引数组。

查看全部Shell变量与函数

1
declare [-fF]
选项 说明
无参数 查询全部Shell变量与函数定义, 与set功能一模一样
-f 查询仅显示函数定义
-F 查询仅显示函数名字

test测试命令

Shell中的 test 命令用于检查某个条件是否成立,它可以进行数值、字符和文件三个方面的测试。

整数比较测试

1
2
3
4
if test 数字1 options 数字2 
then
...
fi

options具体如下:

参数 说明
-eq 等于则为真
-ne 不等于则为真
-gt 大于则为真
-ge 大于等于则为真
-lt 小于则为真
-le 小于等于则为真

演示:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
#!/bin/bash
num1=1 num2=1 num3=2
echo "num1=${num1},num2=${num2},num3=${num3}"

if test $num1 -eq $num2
then
echo 'num1和num2两个数相等!'
else
echo 'num1和num2两个数不相等!'
fi

if test $num2 -eq $num3
then
echo 'num2和num3两个数相等!'
else
echo 'num2和num3两个数不相等!'
fi

字符串比较测试

options具体如下:

参数 说明
= 或 == 等于, 等于返回0代表成功,否则返回1代表失败
!= 不等于
\< 小于
\> 大于
-z 字符串 字符串的长度为零则为真
-n 字符串 字符串的长度不为零则为真

演示:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
#!/bin/bash

str1="hello" str2="world" str3=""
echo "str1=${str1},str2=${str2},str3=${str3}"

if test $str1 = $str2
then
echo 'str1和str2两个字符串相等'
else
echo 'str1和str2两个字符串不相等'
fi

if test $str1 \> $str2
then
echo 'str1大于str2'
else
echo 'str1小于str2'
fi

if test -z $str2
then
echo "str2字符串长度为0"
else
echo "str2字符串长度不为0"
fi

if test -z $str3
then
echo "str3字符串长度为0"
else
echo "str3字符串长度不为0"
fi

文件测试

1
2
3
4
if test options 文件名 
then
...
fi

options具体如下:

参数 说明
-e 文件名 exists, 如果文件存在则为真
-r 文件名 read,如果文件存在且可读则为真
-w 文件名 write,如果文件存在且可写则为真
-x 文件名 execute,如果文件存在且可执行则为真
-s 文件名 string,如果文件存在且至少有一个字符则为真
-d 文件名 directory,如果文件存在且为目录则为真
-f 文件名 file,如果文件存在且为普通文件则为真
-c 文件名 character,如果文件存在且为字符型特殊文件则为真
-b 文件名 如果文件存在且为块特殊文件则为真

演示:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
#!/bin/bash

if test -w ./control1.sh
then
echo '文件已存在并且可写!'
else
echo '文件不存在或不可写!'
fi

if test -e ./control1.sh -a -e ./control2.sh
then
echo '两个文件都存在!'
else
echo '可能有一个或两个文件不存在'
fi

test命令与[]功能一样(更多关于[]的介绍可参考Linux Shell - 运算符)。