参考:
https://www.runoob.com/linux/linux-shell.html
https://www.cnblogs.com/lr-ting/archive/2013/02/28/2936792.html
1 现实问题
1.1 命令行过多
当发现编译内核或者配置内核的命令过长时,可以写一个脚本文件。
#!/bin/bash
KERNEL_DEFCONFIG=rockchip_defconfig
KERNEL_DTS=rk3326-xq-mipi
make ARCH=arm64 $KERNEL_DEFCONFIG && make ARCH=arm64 $KERNEL_DTS.img -j8
if [ $? -eq 0 ]; then
echo "Build kernel ok!"
else
echo "Build kernel failed!"
exit 1
fi
1.2 小结
-
问题
1
-
现象
# !/bin/bash: 坏的解释器: 没有那个文件或目录 问题
-
解决办法
# 因为是在windows上使用 Notepad++ 编辑的文件,所以需要设置 # 1.编码 -> 使用UTF-8编码 # 2.编辑 -> 文档格式转换,转换为UNIX格式
-
2 常用 shell
2.1 基础
-
指定执行脚本的解释器
# !/bin/bash
-
在脚本文件开头添加命令,执行脚本的时候如果遇到返回值为非零,整个脚本就会立即退出
set -e
-
在终端上显示字符串
# -n 不换行 # 可以用''代替 echo "good bash"
-
几个特殊字符
$# : 传递到脚本或函数的参数个数 $* : 以一个单字符串显示所有向脚本传递的参数 $$ : 脚本运行的当前进程 ID 号 $! : 后台运行的最后一个进程的 ID 号 $@ : 与 $* 相同,但是使用时加引号,并在引号中返回每个参数 $- : 显示 shell 使用的当前选项,与 set 命令功能相同 $? : 显示最后命令的退出状态。0 表示没有错误,其他任何值表明有错误
-
比较数值
# -eq 等于则为真 # -ne 不等于则为真 # -gt 大于则为真 # -ge 大于等于则为真 # -lt 小于则为真 # -le 小于等于则为真 num=10 if [ $num -gt 6 ]; then echo '$num > 6' else echo '$num < 6' fi
2.2 sed
-
显示文件指定行内容
# 显示第一行,然后以:进行分割,取第二部分 sed -n '1p' /mnt/mtd/version | cut -d: -f2 # 查找version:所在的行,然后以:进行分割,取第二部分 cat /mnt/mtd/version | grep "version:" | cut -d: -f2
-
修改指定行内容
# 修改第一行的内容 sed -i "1c version:2" version
2.3 cut
-
以字节为单位进行分割
$ who relax pts/0 2021-12-31 11:21 (10.10.110.179) relax pts/1 2021-12-31 08:51 (10.10.110.179) # 取第三个字节 $ who | cut -b 3 l l
-
自定义分隔符分割
$ who relax pts/0 2021-12-31 11:21 (10.10.110.179) relax pts/1 2021-12-31 08:51 (10.10.110.179) # 以 / 进行分割,取第二区域 $ who | cut -d/ -f2 0 2021-12-31 11:21 (10.10.110.179) 1 2021-12-31 08:51 (10.10.110.179)
-
实际使用
# 在文件/mnt/mtd/version里查找version:所在的行,然后以:进行分割,取第二部分 cat /mnt/mtd/version | grep "version:" | cut -d: -f2
2.4 test
命令
2.4.1 数值
-
条件
# -eq 等于则为真 # -ne 不等于则为真 # -gt 大于则为真 # -ge 大于等于则为真 # -lt 小于则为真 # -le 小于等于则为真
-
示例
#!/bin/bash num1=100 num2=100 if test $[num1] -eq $[num2] then echo '两个数相等!' else echo '两个数不相等!' fi
2.4.2 字符串
-
条件
# = 等于则为真 # != 不相等则为真 # -z 字符串 字符串的长度为零则为真 # -n 字符串 字符串的长度不为零则为真
-
示例
#!/bin/bash str1="relax" str2="relaxx" if test $str1 = $str2 then echo '两个字符串相等!' else echo '两个字符串不相等!' fi
2.4.3 文件
-
条件
# -e 文件名 文件存在则为真 # -r 文件名 文件存在且可读则为真 # -w 文件名 文件存在且可写则为真 # -x 文件名 文件存在且可执行则为真 # -s 文件名 文件存在且至少有一个字符则为真 # -d 文件名 文件存在且为目录则为真 # -f 文件名 文件存在且为普通文件则为真 # -c 文件名 文件存在且为字符型特殊文件则为真 # -b 文件名 文件存在且为块特殊文件则为真
-
示例
#!/bin/bash if test -e ./test.sh then echo '文件已存在!' else echo '文件不存在!' fi
2.5 echo
显示内容带颜色
2.5.1 示例
-
shell
脚本中echo
显示内容带颜色显示,echo
显示带颜色,需要使用参数-e
。格式如下:echo -e "\033[字背景颜色;文字颜色m字符串\033[0m"
-
示例
echo -e "\033[41;36m something here \033[0m" # 其中 41 的位置代表底色,36 的位置是代表字的颜色 # 1.字背景颜色和文字颜色之间是英文的 "" # 2.文字颜色后面有个 m # 3.字符串前后可以没有空格,如果有的话,输出也是同样有空格 echo -e "\033[31m 红色字 \033[0m" echo -e "\033[34m 黄色字 \033[0m" echo -e "\033[41;33m 红底黄字 \033[0m" echo -e "\033[41;37m 红底白字 \033[0m"
2.5.2 字体颜色,背景色
-
字体颜色:
30~37
echo -e "\033[30m 黑色字 \033[0m" echo -e "\033[31m 红色字 \033[0m" echo -e "\033[32m 绿色字 \033[0m" echo -e "\033[33m 黄色字 \033[0m" echo -e "\033[34m 蓝色字 \033[0m" echo -e "\033[35m 紫色字 \033[0m" echo -e "\033[36m 天蓝字 \033[0m" echo -e "\033[37m 白色字 \033[0m"
-
背景色:
40~47
echo -e "\033[40;37m 黑底白字 \033[0m" echo -e "\033[41;37m 红底白字 \033[0m" echo -e "\033[42;37m 绿底白字 \033[0m" echo -e "\033[43;37m 黄底白字 \033[0m" echo -e "\033[44;37m 蓝底白字 \033[0m" echo -e "\033[45;37m 紫底白字 \033[0m" echo -e "\033[46;37m 天蓝底白字 \033[0m" echo -e "\033[47;30m 白底黑字 \033[0m"
-
最后面控制选项说明
\33[0m : 关闭所有属性 \33[1m : 设置高亮度 \33[4m : 下划线 \33[5m : 闪烁 \33[7m : 反显 \33[8m : 消隐 \33[30m - \33[37m : 设置前景色 \33[40m - \33[47m : 设置背景色 \33[nA : 光标上移n行 \33[nB : 光标下移n行 \33[nC : 光标右移n行 \33[nD : 光标左移n行 \33[y;xH : 设置光标位置 \33[2J : 清屏 \33[K : 清除从光标到行尾的内容 \33[s : 保存光标位置 \33[u : 恢复光标位置 \33[?25l : 隐藏光标 \33[?25h : 显示光标
2.6 awk
-
命令格式
awk '模式' file awk '{动作}' file awk '模式{动作}' file
-
常用示例
-
查看日期
$ date Tue Nov 15 05:19:31 UTC 2022 $ date | awk '{print $1}' # date 产生的结果就是 file Tue
-
产看网关
$ route Kernel IP routing table Destination Gateway Genmask Flags Metric Ref Use Iface default 172.17.0.1 0.0.0.0 UG 0 0 0 eth0 172.17.0.0 * 255.255.0.0 U 0 0 0 eth0 $ route | awk '$1=="default"{print $2}' 172.17.0.1
-
3 流程控制
3.1 if
-
格式
if condition then command1 command2 ... commandN fi
-
示例
# !/bin/bash a=10 b=20 if [ $a -ne $b ] then echo "a 不等于 b" fi
3.2 if else
-
格式
if condition then command1 command2 ... commandN else command fi
-
示例
# !/bin/bash a=10 b=20 if [ $a == $b ]; then echo "a > b" else [ $a -lt $b ] echo "a < b" fi
3.3 if else-if else
-
示例
if condition1 then command1 elif condition2 then command2 else commandN fi
-
示例
# !/bin/bash a=10 b=20 if [ $a == $b ]; then echo "a = b" elif [ $a -gt $b ]; then echo "a > b" elif [ $a -lt $b ]; then echo "a < b" else echo "other" fi
3.4 for
-
格式
for var in item1 item2 ... itemN do command1 command2 ... commandN done
-
示例
# !/bin/bash for val in 1 2 3 4 5 do echo "The value is: $val" done
3.5 while
-
格式
while condition do command done
-
示例
# !/bin/bash val=1 while(( $val<=5 )) do echo $val let "val++" done
3.6 无限循环(while
)
-
格式
while true do command done
-
示例
# !/bin/bash while true do echo "relax" sleep 1 done
3.7 until
-
格式
until condition do command done
-
示例
#!/bin/bash val=0 # 直到val不小于10为止 until [ ! $val -lt 10 ] do echo $val val=`expr $val + 1` done
3.8 case ... esac
-
格式
case 值 in 模式1) command1 command2 ... commandN ;; 模式2) command1 command2 ... commandN ;; esac
-
示例
#!/bin/bash echo '输入 1 到 4 之间的数字:' echo '你输入的数字为:' read Num # 取值可以为变量或常数,字符串 case $Num in 1) echo '你选择了 1' ;; 2) echo '你选择了 2' ;; 3) echo '你选择了 3' ;; 4) echo '你选择了 4' ;; *) echo '你没有输入 1 到 4 之间的数字' ;; esac
3.9 跳出循环
-
break
#!/bin/bash while true do echo -n "输入 1 到 5 之间的数字:" read aNum case $Num in 1|2|3|4|5) echo "你输入的数字为 $Num!" ;; *) echo "你输入的数字不是 1 到 5 之间的! 游戏结束" break ;; esac done
-
continue
#!/bin/bash while true do echo -n "输入 1 到 5 之间的数字: " read aNum case $Num in 1|2|3|4|5) echo "你输入的数字为 $Num!" ;; *) echo "你输入的数字不是 1 到 5 之间的!" continue echo "游戏结束" ;; esac done
4 函数
4.1 格式
# 可以带function关键字定义fun,也可以直接定义fun,不带任何参数。
# 参数返回,可以显示加:return 返回,如果不加,将以最后一条命令运行结果,作为返回值。 return后跟数值n(0-255)
# 所有函数在使用前必须定义。这意味着必须将函数放在脚本开始部分,直至shell解释器首次发现它时,才可以使用。调用函数仅使用其函数名即可。
[ function ] funname [()]
{
action;
[return int;]
}
4.2 示例
4.2.1 普通函数
-
源码
# !/bin/bash demoFun() { echo "This is my first shell function!" } echo "-----start-----" demoFun echo "------end------"
-
结果
-----start----- This is my first shell function! ------end------
4.2.2 return
函数
-
源码
# !/bin/bash funWithReturn() { echo "这个函数会对输入的两个数字进行相加运算..." echo "输入第一个数字: " read aNum echo "输入第二个数字: " read anotherNum echo "两个数字分别为 $aNum 和 $anotherNum !" return $(($aNum+$anotherNum)) } funWithReturn echo "输入的两个数字之和为 $? !" # 函数返回值在调用该函数后通过 $? 来获得。
-
结果
这个函数会对输入的两个数字进行相加运算... 输入第一个数字: 1 输入第二个数字: 2 两个数字分别为 1 和 2 ! 输入的两个数字之和为 3 !
4.2.3 参数函数
-
源码
# !/bin/bash # 在函数体内部,通过 $n 的形式来获取参数的值,例如,$1表示第一个参数,$2表示第二个参数... # $10 不能获取第十个参数,获取第十个参数需要${10}。当n>=10时,需要使用${n}来获取参数。 funWithParam() { echo "第一个参数为 $1 !" echo "第二个参数为 $2 !" echo "第十个参数为 $10 !" echo "第十个参数为 ${10} !" echo "第十一个参数为 ${11} !" echo "参数总数有 $# 个!" echo "作为一个字符串输出所有参数 $* !" } funWithParam 1 2 3 4 5 6 7 8 9 34 73
-
结果
第一个参数为 1 ! 第二个参数为 2 ! 第十个参数为 10 ! 第十个参数为 34 ! 第十一个参数为 73 ! 参数总数有 11 个! 作为一个字符串输出所有参数 1 2 3 4 5 6 7 8 9 34 73 !