zoukankan      html  css  js  c++  java
  • shell条件选择、循环判断

    一、条件选择、判断

    1、条件选择if语句

    (1)多分支判断

    if 判断条件 1 ; then
      条件为真的分支代码
    elif 判断条件 2 ; then
      条件为真的分支代码
    elif 判断条件 3 ; then
      条件为真的分支代码
    else
      以上条件都为假的分支代码
    fi
    

    逐条件进行判断,第一次遇为“真”条件时,执行其分支,而后结束整个if。

    (2)经典案例:

     #!/bin/bash
    read -p "请输入你的年龄: " age
    
    if [[ $age =~ [^0-9] ]] ;then
    
        echo "输入不合规,请输入整数"
        exit 10
    ### exit命令是Shell内建命令,用于退出当前Shell进程。0为正常退出,非0为异常
    
    elif [ $age -ge 120 ];then
        echo "你已经活到人类极限!"
        exit 20
    
    elif [ $age -ge 60 -a $age -lt 90  ];then
        echo "你已到退休年纪!好好修养!"
        exit 20
    
    elif ((  "$age"  <= "18" ));then
        echo "你未成年,好好学习!"
    
    else
        echo "你已成年,好好工作!"
    
    fi
    

    注意:(( "$age" <= "18" ))、[ -a -o !]、[[]]三者理解有待整理

    https://blog.csdn.net/u010164190/article/details/78973383

    #判断分数
    #!/bin/bash
    read -p "Please input your score: " score
    if [[ $score =~ [^0-9] ]] ;then
        echo "please input a int"
        exit 10
    elif [ $score -gt 100 ];then
        echo "Your score is wrong"
        exit 20
    elif [ $score -ge 85 ];then
        echo "Your score is very good"
    elif [ $score -ge 60 ];then
        echo "Your score is soso"
    else
        echo "You are loser"
    fi
    

    分析:请输入成绩,先判断输入的是否含有除数字以外的字符,有,就报错;没有,继续判断是否大于100,是否大于85,是否大于60。

    cat entrypoint.sh
    #!/bin/sh
    set -ex
    if [ ! $SERVER_IP ]  ##不存在为真,为真即输出
        then
    echo "Please use $SERVER_IP set the IP address of the need to monitor."
        exit 1
    elif [ ! $DHCP_RANGE ]
        then
    echo "Please use $DHCP_RANGE set up DHCP network segment."
        exit 1
    elif [ ! $NEXT_SERVER ]
        then
    echo "Please use $NEXT_SERVER set TFTP PXE booting ."
        exit 1
    elif [ ! $ROOT_PASSWORD ]
        then
    echo "Please use $ROOT_PASSWORD set the root password."
        exit 1
    elif [ ! $DHCP_SUBNET ]
        then
    echo "Please use $DHCP_SUBNET set the dhcp subnet."
        exit 1
    elif [ ! $DHCP_ROUTER ]
        then
    echo "Please use $DHCP_ROUTER set the dhcp router."
        exit 1
    elif [ ! $DHCP_DNS ]
        then
    echo "Please use $DHCP_DNS set the dhcp dns."
        exit 1
    elif [ ! $COBBLER_MASTER ]
        then
    echo "Please use $COBBLER_MASTER set the cobbler master to rsync."
        exit 1
    else
    PASSWORD=`openssl passwd -1 -salt hLGoLIZR $ROOT_PASSWORD`
    sed -i "s/^server: 127.0.0.1/server: $SERVER_IP/g" /etc/cobbler/settings
    sed -i "s/^next_server: 127.0.0.1/next_server: $NEXT_SERVER/g" /etc/cobbler/settings
    sed -i 's/pxe_just_once: 0/pxe_just_once: 1/g' /etc/cobbler/settings
    sed -i 's/manage_dhcp: 0/manage_dhcp: 1/g' /etc/cobbler/settings
    sed -i "s#^default_password.*#default_password_crypted: "$PASSWORD"#g" /etc/cobbler/settings
    sed -i 's/$pxe_menu_items//' /etc/cobbler/pxe/pxedefault.template
    sed -i "s/192.168.1.0/$DHCP_SUBNET/" /etc/cobbler/dhcp.template
    sed -i "s/192.168.1.5/$DHCP_ROUTER/" /etc/cobbler/dhcp.template
    sed -i "s/192.168.1.1;/$DHCP_DNS;/" /etc/cobbler/dhcp.template
    sed -i "s/192.168.1.100 192.168.1.254/$DHCP_RANGE/" /etc/cobbler/dhcp.template
    sed -i "s/^#ServerName www.example.com:80/ServerName localhost:80/" /etc/httpd/conf/httpd.conf
    sed -i "s/service %s restart/supervisorctl restart %s/g" /usr/lib/python2.7/site-packages/cobbler/modules/sync_post_restart_services.py
    rm -rf /run/httpd/*
    apachectl
    cobblerd
    cobbler sync
    cobbler replicate --master=$COBBLER_MASTER --distros=* --profiles=*
    pkill cobblerd
    pkill httpd
    rm -rf /run/httpd/*
    exec supervisord -n -c /etc/supervisord.conf
    fi
    

    2、条件判断 case

    (1)用法格式

    case $name in;
    PART1)
      cmd
      ;;
    PART2)
      cmd
      ;;
    *)
      cmd
      ;;
    esac
    

    注意:case 支持glob 风格的通配符:

      *: 任意长度任意字符
      ?: 任意单个字符
      [] :指定范围内的任意单个字符
      a|b: a 或 b
    

    (2)案例:

    #判断yes or no
    
    #!/bin/bash
    read -p "Please input yes or no: " anw
    case $anw in
    [yY][eE][sS]|[yY])
        echo yes
        ;;
    [nN][oO]|[nN])
        echo no
        ;;
    *)
        echo false
        ;;
    esac
    

    二、四个循环

    1、for

    (1)用法格式

    ① for name  in 列表 ;do
      循环体
    done
    
    ② for (( exp1; exp2; exp3 )) ;do
      cmd
    done
    

    (2)案例1

    #求出(1+2+...+n)的总和
    sum=0
    read -p "Please input a positive integer: " num
    if [[ $num =~ [^0-9] ]] ;then
        echo "input error"
    elif [[ $num -eq 0 ]] ;then
        echo "input error"
    else
        for i in `seq 1 $num` ;do
            sum=$[$sum+$i]
        done
        echo $sum
    fi
    unset zhi  
    

    分析:sum初始值为0,请输入一个数,先判断输入的是否含有除数字以外的字符,有,就报错;没有判断是否为0,不为0进入for循环,i的范围为1~输入的数,每次的循环为sum=sum+i,循环结束,最后输出sum的值。

    #求出(1+2+...+100)的总和
    
    for (( i=1,num=0;i<=100;i++ ));do
            [ $[i%2] -eq 1 ] && let sum+=i
    done
    echo sum=$sum
     分析:i=1,num=0;当i<=100,进入循环,若i÷2取余=1,则sum=sum+i,i=i+1。
    

    案例2:背景,100w文本数据,拆分成200个(docker不能执行太长的数据量),for循环去分别执行200个文本,输出200个结果

    for ((i=0;i<num;i++))

    #!/bin/bash
    
    i=0
    num=200
    
    for ((i=0;i<num;i++))
            {
    sql="select pi.person_name,
           xxx,
           xxx,
           xxx,
           xxx,
           xxx
    from xxx as pi
    where xxx in 
    (`cat 100w_$i`)"
    
    docker exec -it postgres sh -c "export PGPASSWORD=xxx;  psql -h 127.0.0.1 -d database -U xxx -c "COPY (${sql}) to stdout (FORMAT CSV, HEADER);" | tee /tmp/zjz-$i " && docker cp postgres:/tmp/zjz-$i  /mnt/hdd1/zjz/result_$i
    }
    

    2、while

    (1)用法格式

    while 循环控制条件 ;do
      循环
    done
    

    循环控制条件;进入循环之前,先做一次判断;每一次循环之后会再次做判断;条件为“true” ,则执行一次循环;直到条件测试状态为“false” 终止循环

    (2)特殊用法(遍历文件的每一行):

    while read line; do控制变量初始化
      循环体
    done < /PATH/FROM/SOMEFILE
    或cat /PATH/FROM/SOMEFILE | while read line; do
      循环体
    done
    

    依次读取/PATH/FROM/SOMEFILE文件中的每一行,且将行赋值给变量line

    (3)案例:

    #100以内所有正奇数之和
    sum=0
    i=1
    while [ $i -le 100 ] ;do
    if [ $[$i%2] -ne 0 ];then
        let sum+=i
        let i++
    else
        let i++
    fi
    done
    echo "sum is $sum"
    

    分析:sum初始值为0,i的初始值为1;请输入一个数,先判断输入的是否含有除数字以外的字符,有,就报错;没有当i<100时,进入循环,判断 i÷2取余 是否不为0,不为0时为奇数,sum=sum+i,i+1,为0,i+1;循环结束,最后输出sum的值。

    while常见形式:(重定向、循环)

    while read line
    do
           …
    done < file
    

    实战案例(一个txt文本,两行为一组,有四列逗号相隔的文件,处理后如下)

    1 500xxxxxxxxxxxxxxx 陈宕 R5xxxxxxxxxxxxxx7110018
    1 500xxxxxxxxxxxxxxxx 陈铭 R500xxxxxxxxxxxx8070005
    2 5ddllllllllll0036 李业盛 Rdsjdjkdsvlsdvlsdkv069
    2 5ddllllllllll0036 李业盛 Rdsjdjkdsvlsdvlsdkv069
    3 6ddllllllllll0036 张业 Rdsjdjkdsvlsdvlsdkv069
    3 6ddllllllllll0036 张业 Rdsjdjkdsvlsdvlsdkv069
    

    需求:

    1、以第一列为序号建分组,如第x组;

    2、以2、3、4列建下级子目录500xxxxxxxxxxxxxxx_陈宕_R5xxxxxxxxxxxxxx7110018;

    3、将与第四列同名的*.bmp文件移到对应的目录下;

    4、不成功的行,做标记。

    #!/bin/bash
    echo > ./info.log
    rm -rf 第*
    num=0
    num_info=0
    is_no_num=0
    
    ########a b c d 分别代表txt文本中的四列数据
    
    while read a b c d   #####  while 读取每一行以空格为分界线
    
    do
           let num_info=${num_info}+1
    
            if [ $a -eq $num ];then
                    mkdir /root/group2/第${a}组
                    mkdir /root/group2/第${a}组/$info
                    mkdir /root/group2/第${a}组/${b}_${c}_${d}
                    mv /root/group2/${d}.bmp /root/group2/第${a}组/${b}_${c}_${d}
                    mv /root/group2/${d_up}.bmp /root/group2/第${a}组/$info
    
            else
    
                    let is_no=$is_no_num+1
                    if [ $is_no -eq $num_info ];then
                            echo  $is_no_num >>./info.log   ###没成功的行号做标记
                    fi
    
                    is_no_num=$num_info
                    num=$a
                    info=${b}_${c}_${d}
                    d_up=$d
                    continue
            fi
    
    done < output.txt
    

    3、until 循环

    (1)用法

    unitl 循环条件 ;do
      循环
    done
    

    进入条件:循环条件为true ;退出条件:循环条件为false;刚好和while相反,所以不常用,用while就行。

    (2)案例

    #监控xiaoming用户,登录就杀死
    until pgrep -u xiaoming &> /dev/null ;do
            sleep 0.5
    done
    pkill -9 -u xiaoming
    

    分析:每隔0.5秒扫描,直到发现xiaoming用户登录,杀死这个进程,退出脚本,用于监控用户登录。

    4、select 循环与菜单

    (1)用法

    select variable in list
    do
      循环体命令
    done
    
    • ① select 循环主要用于创建菜单,按数字顺序排列的示菜单项将显示在标准错误上,并显示PS3 提示符,等待用户输入
    • ② 用户输入菜单列表中的某个数字,执行相应的命令
    • ③ 用户输入被保存在内置变量 REPLY 中
    • ④ select 是个无限循环,因此要记住用 break 命令退出循环,或用 exit 按 命令终止脚本。也可以按 ctrl+c退出循环
    • ⑤ select 和 经常和 case 联合使用
    • ⑥ 与for循环类似,可以省略 in list, 此时使用位置参量

    (2)示例

    #生成菜单,并显示选中的价钱
    PS3="Please choose the menu: "
    select menu in mifan huimian jiaozi babaozhou quit
    do
            case $REPLY in
            1|4)
                    echo "the price is 15"
                    ;;
            2|3)
                    echo "the price is 20"
                    ;;
            5)
                    break
                    ;;
            *)
                    echo "no the option"
            esac
    done
    
    
    #!/bin/bash
    echo "What is your favourite OS?"
    select name in "Linux" "Windows" "Mac OS" "UNIX" "Android"
    do
        case $name in
            "Linux")
                echo "Linux是一个类UNIX操作系统,它开源免费,运行在各种服务器设备和嵌入式设备。"
                break
                ;;
            "Windows")
                echo "Windows是微软开发的个人电脑操作系统,它是闭源收费的。"
                break
                ;;
            "Mac OS")
                echo "Mac OS是苹果公司基于UNIX开发的一款图形界面操作系统,只能运行与苹果提供的硬件之上。"
                break
                ;;
            "UNIX")
                echo "UNIX是操作系统的开山鼻祖,现在已经逐渐退出历史舞台,只应用在特殊场合。"
                break
                ;;
            "Android")
                echo "Android是由Google开发的手机操作系统,目前已经占据了70%的市场份额。"
                break
                ;;
            *)
                echo "输入错误,请重新输入"
        esac
    done
    

    分析:PS3是select的提示符,自动生成菜单,选择5  break退出循环。

    三、循环里的一些用法

    1、循环控制语句

    (1)语法

    continue [N]:提前结束第N层的本轮循环,而直接进入下一轮判断;最内层为第1层 break [N]:提前结束第N层循环,最内侧为第1层

     例:while CONDTITON1; do
      CMD1
    if CONDITION2; then
      continue / break
    fi
      CMD2
    done
    

    (2)案例:

    #①求(1+3+...+49+53+...+100)的和
    
    #!/bin/bash
    sum=0
    for i in {1..100} ;do
            [ $i -eq 51 ] && continue
            [ $[$i%2] -eq 1 ] && { let sum+=i;let i++; }
    done
    echo sum=$sum
    

    分析:做1+2+...+100的循环,当i=51时,跳过这次循环,但是继续整个循环,结果为:sum=2449

    #②求(1+3+...+49)的和
    
    #!/bin/bash
    sum=0
    for i in {1..100} ;do
            [ $i -eq 51 ] && break
            [ $[$i%2] -eq 1 ] && { let sum+=i;let i++; }
    done
    echo sum=$sum
    

    分析:做1+2+...+100的循环,当i=51时,跳出整个循环,结果为:sum=625

    2、循环控制shift命令

    (1)作用

    用于将参数列表list左移指定次数,最左端的那个参数就从列表中删除,其后边的参数继续进入循环

    (2)案例:

    #①创建指定的多个用户
    
    #!/binbash
    if [ $# -eq 0 ] ;then
            echo "Please input a arg(eg:`basename $0` arg1)"
            exit 1
    else
            while [ -n "$1" ];do
                    useradd $1 &> /dev/null
                    shift
            done
    fi
    

    分析:如果没有输入参数(参数的总数为0),提示错误并退出;反之,进入循环;若第一个参数不为空字符,则创建以第一个参数为名的用户,并移除第一个参数,将紧跟的参数左移作为第一个参数,直到没有第一个参数,退出。

    #②打印直角三角形的字符
    
    #!/binbash
    while (( $# > 0 ))
    do
            echo "$*"
            shift
    done
    

    3、返回值结果

    true 永远返回成功结果
    : null command ,什么也不干,返回成功结果
    false 永远返回错误结果
    

    创建无限循环

    while true ;do
      循环体
    done
    

    4、循环中可并行执行,使脚本运行更快

    (1)用法

    for name in 列表 ;do
      {
      循环体
      }&
    done
    wait
    

    (2)实例:

    #搜寻自己指定ip(子网掩码为24的)的网段中,UP的ip地址
    read -p "Please input network (eg:192.168.0.0): " net
    echo $net |egrep -o "<(([0-9]|[1-9][0-9]|1[0-9]{2}|2[0-4][0-9]|25[0-5]).){3}([0-9]|[1-9][0-9]|1[0-9]{2}|2[0-4][0-9]|25[0-5])>"
    [ $? -eq 0 ] || ( echo "input error";exit 10 )
    IP=`echo $net |egrep -o "^([0-9]{1,3}.){3}"`
    for i in {1..254};do
            {
            ping -c 1 -w 1 $IP$i &> /dev/null && 
            echo "$IP$i is up" 
            }&
    
    done
    wait
    

    分析:请输入一个IP地址例192.168.37.234,如果格式不是0.0.0.0 则报错退出;正确则进入循环,IP变量的值为192.168.37.  i的范围为1-254,并行ping 192.168.37.1-154,ping通就输出此IP为UP。直到循环结束。

    四、信号捕获trap

    1、用法格式

    trap ' 触发指令' 信号,自定义进程收到系统发出的指定信号后,将执行触发指令,而不会执行原操作
    trap '' 信号,忽略信号的操作
    trap '-' 信号,恢复原信号的操作
    trap -p,列出自定义信号操作
    

    信号可以3种表达方法:信号的数字2、全名SIGINT、缩写INT

    2、常用信号

    1) SIGHUP: 无须关闭进程而让其重读配置文件
    2) SIGINT: 中止正在运行的进程;相当于Ctrl+c
    3) SIGQUIT: 相当于ctrl+
    9) SIGKILL: 强制杀死正在运行的进程
    15) SIGTERM :终止正在运行的进程(默认为15)
    18) SIGCONT :继续运行
    19) SIGSTOP :后台休眠
    9 信号,强制杀死,捕获不住
    

     3、案例

    #①打印0-9,ctrl+c不能终止
    #!/bin/bash
    trap 'echo press ctrl+c' 2
    for ((i=0;i<10;i++));do
            sleep 1
            echo $i
    done
    

    分析:i=0,当i<10,每休眠1秒,i+1,捕获2信号,并执行echo press ctrl+c

    #②打印0-3,ctrl+c不能终止,3之后恢复,能终止
    
    #!/bin/bash
    trap '' 2
    trap -p
    for ((i=0;i<3;i++));do
            sleep 1
            echo $i
    done
    trap '-' SIGINT
    for ((i=3;i<10;i++));do
            sleep 1
            echo $i
    done
    

    分析:i=0,当i<3,每休眠1秒,i+1,捕获2信号;i>3时,解除捕获2信号。

    六、分享几个有意思的小脚本

    1、、echo打印颜色字

    echo -e "33[31malong33[0m" 显示红色along
    echo -e "33[1;31malong33[0m" 高亮显示红色along
    echo -e "33[41malong33[0m" 显示背景色为红色的along
    echo -e "33[31;5malong33[0m" 显示闪烁的红色along
    color=$[$[RANDOM%7]+31]
    echo -ne "33[1;${color};5m*33[0m" 显示闪烁的随机色along
    

    2、9x9乘法表

    #!/bin/bash
    for a in {1..9};do
            for b in `seq 1 $a`;do
                    let c=$a*$b ;echo -e "${a}x${b}=$c	c"
            done
            echo    
    done
    

    2、彩色等腰三角形

    #!/bin/bash
    read -p "Please input a num: " num
    if [[ $num =~ [^0-9] ]];then
            echo "input error" 
    else
            for i in `seq 1 $num` ;do
                    xing=$[2*$i-1]
                    for j in `seq 1 $[$num-$i]`;do
                            echo -ne " "
                    done
                    for k in `seq 1 $xing`;do
                            color=$[$[RANDOM%7]+31]
                            echo -ne "33[1;${color};5m*33[0m"
                    done
                    echo
            done
    fi
    

    3、国际象棋棋盘

    #!/bin/bash
    red="33[1;41m  33[0m"
    yellow="33[1;43m  33[0m"
    
    for i in {1..8};do
            if [ $[i%2] -eq 0 ];then
                    for i in {1..4};do
                            echo -e -n "$red$yellow";
                    done
                    echo
            else
                    for i in {1..4};do
                            echo -e -n "$yellow$red";
                    done
                    echo 
            fi
    done
    

    七、生成连续数字(实用)

    1、$ echo  {1..10}

    1 2 3 4 5 6 7 8 9 10
    

    2、 awk 'BEGIN { for (i=0; i<=20; i++) printf("%02d ", i) }'

    00 01 02 03 04 05 06 07 08 09 10 11 12 13 14 15 16 17 18 19 20
    

    3、seq -w 1 10

    01
    02
    03
    04
    05
    06
    07
    08
    09
    10
    

    4、 printf "%02d " {1..10}

    01 02 03 04 05 06 07 08 09 10
    

    5、seq -f "%03g" 0 10

    000
    001
    002
    003
    004
    005
    006
    007
    008
    009
    010
    

    运用1

    for i in `echo {10..20}`;do mkdir  第$i组;done
    
    $ ls
    第10组  第11组  第12组  第13组  第14组  第15组  第16组  第17组  第18组  第19组  第20组
    
    
    
    
  • 相关阅读:
    【调侃】IOC前世今生
    经典 makefile 教程
    Win7  CMD大全
    关于验证码识别3
    关于验证码识别2
    关于验证码识别 1
    DirectUI的初步分析-转
    winform软件版本检测自动升级开发流程(转)
    如何让在panel里的子窗体随panel的大小改变而变化?(转)
    C#如何控制panel加载的字窗体中控件的位置随着显示器分辨率大小而改变
  • 原文地址:https://www.cnblogs.com/zjz20/p/14359411.html
Copyright © 2011-2022 走看看