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

posted @ 2022-08-09 14:25  lynalee  阅读(685)  评论(0编辑  收藏  举报