shell 常用语法
1 常规语法
1.1 变量赋值等号的两边不能有空格
correct:
var1=1 var2="hello world" var3=${var2} # 变量的引用,在不会引起歧义的情况下大括号可以省略
false:
var1 = 1 var2 = "hello world"
1.2 if 判断的条件的中括号两边必须有空格
correct:
if [ ! -d "${LOCAL_PATH}" ]; then echo "no such directory: ${LOCAL_PATH}" fi
false:略
1.3 if,for,while语法要点
1.3.1 if elif else fi
if [ -d ${dir_now_full} ];then LOCAL_DIR=${dir_now_full} if [ -d ${dir_yes_full} ];then mv ${dir_yes_full}/*.txt ${dir_now_full} echo "dir_yes_full=${dir_yes_full}" rm -rf ${dir_yes_full} fi elif [ -d ${dir_yes_full} ];then LOCAL_DIR=${dir_yes_full} else echo "no dir!" exit 1 fi
判断字符串是否包含
if [[ "${array[$i]}" =~ "-" ]];then 。。。 fi
1.3.2 for do done
for 有好几种语法,例如 for ... in do done ; 还有和C类似的写法:for ((i=0;i<10;i++)),这里只介绍一种,虽写起来麻烦,但笔者认为这种最为通用,前面的语法都有自己的局限性。
for i in ${!array[@]} # i为元素下标 do echo "${array[$i]}" # 通过元素下标打印列表的各个元素 fi done
附录一个for的一行写的代码
for i in seq 0 `expr ${#array1[@]} - 1` ;do array2[$i]=0 ;done
1.3.3 while do done
while true ; do echo $num break done
1.3.4 case in 这里附上一个非常实用的小例子
通过解析命令行参数,执行对应操作
init(){ ARGS=`getopt -o -I:-u:-p:h -l IP:,user:,password:,help -- "$@"` if [ "$?" -ne 0 ]; then message "option error,exit" usage exit 1 fi eval set -- "${ARGS}" TIMEOUT=30 LOGFILE="$HOME/putfile.log" while true ; do case "$1" in -I|--IP) IP="$2" ; shift ;; -u|--user) USER="$2" ; shift ;; -p|--password) PASSWORD="$2" ; shift ;; -h|--help) usage ; exit 0 ;; --) shift ; break ;; *) echo "option error! exit!" exit 1 ;; esac shift done }
1.3.5 遍历文件行
while read line do echo $line done < $FILENAME
2 变量操作
2.0 特殊变量
$$ Shell本身的PID $! Shell最后运行的后台Process的PID $? 最后运行的命令的结束代码(返回值) $- 使用Set命令设定的Flag一览 $* 所有参数列表。如"$*"用双引号括起来的情况、以"$1 $2 … $n"的形式输出所有参数,表示的是用内部分割符 IFS 连接起来的一个完整的统一字符串。 $@ 所有参数列表。如"$@"用双引号括起来的情况、以"$1" "$2" … "$n" 的形式输出所有参数, 仍然表示的是各个输入的参数. 所以这也就解释了, 除非特殊情况, 为什么推荐使用 $@ 而不是 $* 展开参数列表了. $# 添加到Shell的参数个数 $0 Shell本身的文件名 $1~$n 添加到Shell的各参数值。$1是第1参数、$2是第2参数…。
2.1 变量赋值
var1="Hello world" var2=$var1 # 此时var2的值也为"Hello world" var3=$(echo $var1) # $() 中的为可单独执行的命令,此处将echo $var1的标准输出赋值给了var3,var3为"Hello world" var4=·echo $var1· # ··的作用与$()相同
在shell脚本中,所有的标量如果没有进行特殊申明,则都为全局变量,在函数中可以任意使用
2.2 变量截取
[root@gluster-node2 server]# echo ${SERVICE_NAME} LY-nalee-1 [root@localhost ~]# echo ${SERVICE_NAME#*-} nalee-1 [root@gluster-node2 server]# echo ${SERVICE_NAME##*-} 1 [root@localhost ~]# echo ${SERVICE_NAME%-*} LY-nalee [root@gluster-node2 server]# echo ${SERVICE_NAME:1} Y-nalee-1 [root@gluster-node2 server]# echo ${SERVICE_NAME:1:3} Y-n [root@gluster-node2 server]# echo ${SERVICE_NAME/LY-/ly} lynalee-1 [root@gluster-node2 server]# echo ${SERVICE_NAME//e/E} LY-nalEE-1 [root@gluster-node2 server]# echo ${SERVICE_NAME: -1} # 注意-1前有空格 1
2.3 变量替换
[root@gluster-node2 server]# echo $var abc-12345-abc [root@gluster-node2 server]# echo ${var/abc/ABC} ABC-12345-abc [root@gluster-node2 server]# echo ${var/#abc/ABC} ABC-12345-abc [root@gluster-node2 server]# echo ${var/%abc/ABC} abc-12345-ABC
2.4 变量比较
[root@gluster-node2 server]# cat 1.sh if [[ "123abc456" =~ "abc" ]];then echo "\$?=$?" fi [root@gluster-node2 server]# bash 1.sh $?=0
2.5 数组变量
2.5.1 数组长度
${#array_var[@]}
2.5.2 数组拷贝
newarray=(${array[@]})
2.5.3 数组打印
echo ${newprogram[@]} # 会打印出数组的所有元素,如果echo ${newprogram}则只会打印出数组的第一个元素
2.5.4 数组追加
array_name[${#array_name[@]}]=value
2.5.5 数组切片
array=(1 2 3 4 5 6 7) ${array[@]} # 整个数组 ${array[@]:0:2} # 0-1下标的元素
2.6 变量算数计算
2.6.1 加减
$((num1+2))
2.7 变量执行
cmd="sh /root/bin/test.sh 50 7" $cmd # 不需要用$()括起来,直接这样写即可
3 sed实例
3.1 获取匹配的两行之间的内容
cat test.txt | sed -n '/First Line/,/Second Line/p'
3.2 去除空格,提取子串
示例:提取最右侧的数字
示例:提取RouteNum不为0的PortId的值
No: 0 Index: 0 PortId: 1 RouteNum: 5 No: 0 Index: 1 PortId: 1 RouteNum: 3 No: 0 Index: 2 PortId: 1 RouteNum: 8 No: 0 Index: 3 PortId: 1 RouteNum: 0 No: 0 Index: 4 PortId: 1 RouteNum: 2 No: 0 Index: 5 PortId: 1 RouteNum: 22
sed s/[[:space:]]//g | sed 's/.*RouteNum:\([0-9]\+\)/\1/g' # 分两个sed sed 's/.*RouteNum:[[:space:]]*\([0-9]\+\)/\1/g' # 一个sed
3.3 在匹配行上下插入行
sed -i "/match_line/a\after match" test.txt
sed -i "/match_line/i\before match" test.txt
3.4 删除指定两行号之间的内容
sed -i '17,19d' test.ini
3.5 删除匹配行之后的所有内容
sed -i '/\[global\]/,$d' config.ini
3.6 打印匹配行之后的所有内容
sed -n '/\[global\]/,$p' config.ini
4 awk
例一:
查看watchdogd的进程
[root@localhost ~]# ps -elf | grep watchdogd
1 S root 78 2 0 -40 - - 0 - Nov10 ? 00:00:00 [watchdogd] 0 S root 2651499 2651401 0 80 0 - 2302 - 16:51 pts/0 00:00:00 grep --color=auto watchdogd
获得具体的pid
[root@localhost ~]# ps -elf | grep watchdogd | grep -v grep | awk '{print $4}'
78
例二:
用awk进行大小比较,判断磁盘使用率是否超过了50%,如果超过了50%就返回1,否则为0
awk -v n1=${DiskUsage%?} -v n2=50 'BEGIN{print(n1>n2)?1:0}'
这里的“%?”参考之前的变量语法,为去除变量后面的百分号
例三:统计字段个数
[root@localhost ~]# echo "string=123,dsafdsa,546" | awk -F ',' '{print NF}' 3
5 一些shell工具的技巧及使用
5.1 top不交互的打印
top的坑,不用-b,sed就失效了,这里获取了空闲cpu
top -n 1
top -n 1 -b | grep '%Cpu(s)' | sed 's/.* \([0-9]\+.[0-9]\+\) id.*/\1/g'
5.2 rpm
rpm -Uvh *.rpm --nodeps --force # 强制安装当前目录下的所有rpm包 rpm -ivh XXX # 安装XXX包 rmp -qa | grep XXX # 查看已安装的包 rpm -e --nodeps xxx # 卸载xxx包 error: rpmdb: BDB0113 Thread/process 1654361/139711472111104 failed: BDB1507 Thread died in Berkeley DB library 报错的解决方法 [root@localhost]# cd /var/lib/rpm [root@localhost rpm]# rm -rf __db* [root@localhost rpm]# rpm --rebuilddb
5.3 yum
yum install --downloadonly --downloaddir=./gcc gcc gcc-c++ # yum只下载,不安装 yum reinstall --downloadonly --downloaddir=./gcc gcc gcc-c++ # yum只下载,不安装
yum deplist perf # 查看依赖
yum -y install yum-utils # 安装yum的扩展工具,然后可进行下面的操作 repotrack ansible # 下载ansible的全部依赖包
5.4 service
查看环境上已安装的所有的service
方法1:
chkconfig --list
方法2:
ls /etc/init.d
5.5 mount
[root@gfs3 data]# umount /application/gfsmount [root@gfs3 data]#
5.6 查看用户密码相关信息
[root@localhost /]# chage -l root Last password change : never Password expires : never Password inactive : never Account expires : never Minimum number of days between password change : 0 Maximum number of days between password change : 90 Number of days of warning before password expires : 7
5.7 date打印好看的年月日
[root@localhost] # date '+%y-%m-%d %H:%M:%S' 22-10-30 12:35:26 LOG_FILE='./logfile' gen_log() { LOG_TIME=$(date +%Y%m%d%H%M%S) echo "${LOG_TIME} $*" | tee -a ${LOG_FILE} }
5.9 read
参考:https://www.cnblogs.com/hjfeng1988/p/10621649.html
local args=$*; IFS_OLD=${IFS}; IFS=,; read var1 var2 <<< $(echo "$*"); IFS=${IFS_OLD}
5.10 seq xargs 快速创建多个文件
seq 100 120 |xargs -i dd if=/dev/zero of=testfile.{} bs=1K count=1
6 特殊变量
$$,$!,...待补充
7 其他shell技巧
7.1 一行代码通过ip查找到对应网卡
ip -br address | grep 192.168.1.111 | awk '{print $1}' | cut -d@ -f1
7.2 上传文件到别的pc
lftp_put_file(){ lftp -u ${USER},${PASSWORD} sftp://${IP}:${PORT} << EOF mkdir -p ${REMOTE_PATH} cd '${REMOTE_PATH}' put ${SOURCEFILE} ${SOURCEFILE} bye EOF }
7.3 看门狗程序
如果子进程执行的时间过长,使用看门狗杀死子进程
timeout(){ #lftp command=$* $command >> $LOGFILE & commandpid=$! #WatchDog Process now_time=$(date +%s) remain_time=$[ ${TIMEOUT}-($now_time-$start_time) ] if [ ${remain_time} -lt 0 ];then remain_time=0 fi ( sleep ${remain_time} ; kill -9 $commandpid > /dev/null 2>&1) & watchdog=$! sleeppid=$PPID logger "put_file pid is:${commandpid}" logger "watchdog pid is:${watchdog}" logger "master pid is:$$" logger "session pid is:${sleeppid}" wait $commandpid > /dev/null 2>&1 ret=$? logger "SFTP execution results:${ret}" if [ $ret -ne 0 ];then logger "SFTP execution failed!" kill -15 -$$ exit 1 fi } start_time=$(date +%s) timeout lftp_put_file logger "success" kill -15 -$$
7.4 远程并跳转用户执行
因为有保护机制,只能登陆远端pc的lynalee用户
获得远端的lynalee_test 容器的 容器id
container_id=`ssh lynalee@192.168.1.12 "sudo su -c 'docker ps | grep lynalee_test'" | head -n 1 | awk '{print $1}'`
7.5 xfs v5文件系统查看文件的创建时间
ls -i ${filepath} # 查看出文件的inode df -T ${filepath} # 查出磁盘分区 xfs_db -r -c "inode 1678183210" -c "p v3.crtime.sec" /dev/mapper/ncl-paasdata
7.6 查看磁盘io和占用最多的进程
iostat -x 1 10 iotop -oP pidstat -d 1
参考的其他作者连接:https://cloud.tencent.com/developer/article/1718267