shell脚本进阶一(for,while,continue,break,select等等)

脚本进阶一

一、for循环的第二种写法:

众所周知,for有两种写法

  • 第一种:for i in k8s-node{1..3};do setenforce 0;done
  • 第二种写法:C语言风格

直接写怎么用:

#for后必须写两个括号,又称双小括号写法
[root@linux1 ~]# cat for_2.sh 
#!/bin/bash

for ((i=1,sum=0;i<=100;i++));do
    let sum+=i
done
echo "sum=${sum}"
[root@linux1 ~]# bash for_2.sh
sum=5050

二、while循环

我喜欢这样写,一直循环然后用break退出

[root@linux1 ~]# cat while_sum.sh
#!/bin/bash
i=1
sum=0
while true;do
    let sum+=i
    let i++
    if [ $i -gt 100 ];then
        break
    fi
done
echo "sum=${sum}"
[root@linux1 ~]# bash  while_sum.sh 
sum=5050

while的高级用法:读取标准输入的内容实现循环

[root@linux1 ~]# cat while_2.sh
#!/bin/bash
while read line
do
    echo $line
done < /etc/fstab
[root@linux1 ~]# bash while_2.sh

#
# /etc/fstab
# Created by anaconda on Thu Aug 8 19:04:39 2019
#
# Accessible filesystems, by reference, are maintained under '/dev/disk'
# See man pages fstab(5), findfs(8), mount(8) and/or blkid(8) for more info
#
/dev/mapper/centos-root / xfs defaults 0 0
UUID=3778e6e0-8f51-4843-8b8f-239c8b5e826b /boot xfs defaults 0 0
/dev/mapper/centos-home /home xfs defaults 0 0
/dev/mapper/centos-swap swap swap defaults 0 0

并且看网友的资料说,上面写法中wile使用重定向机制,fstab文件内容被重定向给了整个while语句,这个特性要注意下

while的注意事项1:管道传递内容:echo "abc xyz" | while read line ;do {};done 众所周知管道会开启子shell,子shell内的数组,变量,函数在函数外部均不生效

#!/bin/bash
echo "abc xyz" | while read line
do
    new_var=$line
done
echo new_var is null: $new_var?

三、until循环

until CONDITION; do  循环体 
done 

进入条件: CONDITION 为false 
退出条件: CONDITION 为true 

四、continue特殊用法

continue [N]:提前结束第N层的本轮循环,最内层为第一层

默认N为1,只退一层循环。比如复杂的代码套了好几层的话,continue 2可以直接提前结束两层的循环,直接进入下一轮判断,continue候面的代码不执行了

五、break用法

break [N]:提前结束第N层循环,最内层为第1层

注意:这个直接结束第N层所有循环了,而continue只是结束第N层的本轮循环

六、shift命令

shift [n]

很重要,脚本中经常用到。将参数列表左移n次

[root@linux1 ~]# cat shift.sh 
#!/bin/bash
#判断脚本参数是否为0,不为0则执行循环
while [ $# -ne 0 ];do
    echo $1  #打印第一个参数
    shift       #所有参数左移,第一个参数被挤走,第二个参数变成第一个参数
done
[root@linux1 ~]# ./shift.sh a b c d f
a
b
c
d
f

七、select循环与菜单

select经常与case一起用;还有就是PS3作为提示符;还有就是要配合break或exit退出循环

[root@linux1 ~]# cat select.sh 
#!/bin/bash
PS3="你想干啥:"
select choice in eating wc sleep quit
do
    case $choice in
        eating)
            echo "you can eat some food now."
            ;;
        wc)
            echo "you can go go to wc now."
            ;;
        sleep)
            echo "you can go to sleep now."
            ;;
        quit)
            exit 0
    esac
done

效果

[root@linux1 ~]# bash select.sh
1) eating
2) wc
3) sleep
4) quit
你想干啥:1
you can eat some food now.
你想干啥:2
you can go go to wc now.
你想干啥:3
you can go to sleep now.
你想干啥:4

八、函数

载入函数:

  • . filename
  • source filename

九、删除函数

可以使用unset删除

比如:一个简单的函数,执行没有问题

[root@linux1 ~]# cat function.sh
#!/bin/bash
hi(){
    echo hi
}
hi
[root@linux1 ~]# bash function.sh
hi

中间加一行unset,就报错了,因为函数被删除了

[root@linux1 ~]# cat function.sh
#!/bin/bash
hi(){
    echo hi
}
unset hi
hi
[root@linux1 ~]# bash function.sh
function.sh: line 6: hi: command not found

还可以通过定义空函数实现,这是我在系统中的脚本中发现的

# ubuntu的/lib/lsb/init-functions
# Pre&Post empty function declaration, to be overriden from /lib/lsb/init-functions.d/*
log_daemon_msg_pre () { :; } 
log_daemon_msg_post () { :; }
log_begin_msg_pre () { :; }
log_begin_msg_post () { :; }
log_end_msg_pre () { :; }
log_end_msg_post () { :; }
log_action_msg_pre () { :; }
log_action_msg_post () { :; }
log_action_begin_msg_pre () { :; }
log_action_begin_msg_post () { :; }
log_action_end_msg_pre () { :; }
log_action_end_msg_post () { :; }

十、函数变量的生存时间

环境变量:当前shell和子shell有效

本地变量:只在当前shell进程中有效,包括脚本函数

局部变量:函数的生命周期,函数结束时变量销毁

局部变量的定义:local AGE=20

十一、函数的递归

联想到fork炸弹

:(){:|:&};:

脚本实现

cat bomb.sh
#!/bin/bash
./$0|./$0&

十二、信号的捕捉trap

看另一篇文章

上一篇:如何检查列表中的值是否存在文件每行中?


下一篇:Linux1:入门概述与虚拟机环境搭建