Shell脚本--循环与函数,数组

循环语句

在实际工作中,经常会遇到某项任务需要多次执行的情况,而每次执行时仅仅是处理的   对象不一样,其他命令相同。例如,根据通讯录中的姓名列表创建系统账号,根据服务器清   单检查各主机的存活状态,根据 IP 地址黑名单设置拒绝访问的防火墙策略等

当面对各种列表重复任务时,使用简单的 if 语句已经难以满足要求,而顺序编写全部代码更是显得异常烦琐、困难重重而循环语句,可以很好地解决类似问题

循环的退出命令

break 退出脚本

continue 退出循环

 for 循环

使用 for 循环语句时,需要指定一个变量及可能的取值列表,针对每个不同的取值重复执行相同的命令序列,直到变量值用完退出循环。在这里,“取值列表”称为 for 语句的执行条件,其中包括多个属性相同的对象,需要预先指定(如通讯录、IP 黑名单) 

for循环结构

//取值列表相当于一个数组,i是数组内每个值

for 变量名 in 取值列表

do

        命令序列

done

 for循环示例

#示例1
for i in $(cat users.txt)    //文本内每一行就是一遍循环,变量i就是每一行内容
do 
    echo "hello $i"
done

#users.txt 内容
a
b
c
d
e
f

#示例2
for i in $(seq 1 10)        //循环10遍,i相当于一个数组,数组里面有10个数,分别是1到10
do
    echo "$i"
done

#示例3
for ((i=1;i<=5;i++))         //定义循环初始值i=1,循环条件i<=5,循环的迭代信息,每次循环后初始值加1 i++
do
    echo "$i"
done 

 while 循环语句

for 循环语句非常适用于列表对象无规律,且列表来源已固定(如某个列表文件)的场合。而对于要求控制循环次数、操作对象按数字顺序编号、按特定条件执行重复操作等情况,   则更适合使用另外一种循环——while 语句

while 语句的结构

使用 while循环语句时,可以根据特定的条件反复执行一个命令序列,直到该条件不再满足时为止。在脚本应用中,应该避免出现死循环的情况,否则后边的命令操作将无法执行。   因此,循环体内的命令序列中应包括修改测试条件的语句,以便在适当的时候使测试条件不再成立,从而结束循环。

while 条件测试操作

do

命令序列

done

while 语句的执行流程

首先判断 while 后的条件测试操作结果,如果条件成立,则执行 do…done 循环体中的命令序列;返回 while 后再次判断条件测试结果,如果条件仍然成立,则继续执行循环体;再次返回到 while 后,判断条件测试结果……如此循环,直到 while 后的条件测试结果不再成立为止,最后跳转到 done 语句,表示结束循环

使用 while 循环语句时,有两个特殊的条件测试操作,即 true(真)和 false(假)。使用 true 作为条件时,表示条件永远成立,循环体内的命令序列将无限执行下去,除非强制终止脚本(或通过 exit 语句退出脚本);反之,若使用 false 作为条件,则循环体将不会被执行。这两个特殊条件也可以用在 if 语句的条件测试中

while循环示例

#!/bin/bash
#循环批量创建用户stu1-stu20
PREFIX="stu"    //定义用户名称的前半部分
i=1            循环初始值
while [ $i -le 20 ]     //循环条件 变量i小于等于20
do    //循环体开始
    useradd${PREFIX}$i    //命令序列,创建用户
    echo"123456" | passwd --stdin ${PREFIX}$i &> /dev/null    //设置密码,将密码设置成功这些提示信息输出到/dev/null黑洞中 
    let i++ / ((i++))         这两个命令都可以执行i+1也就是循环迭代操作
done    //循环体结束

 除了以上两种循环外还有until循环,until循环与while循环基本一样,只不过until循环是循环条件不成立时执行循环,和while正好相反,这里不再赘述

Shell函数

Shell 函数可用于存放一系列的指令。在 Shell 脚本执行的过程中,函数被置于内存中, 每次调用函数时不需要从硬盘读取,因此运行的速度比较快。在 Shell 编程中函数并非是必须的元素,但使用函数可以对程序进行更好的组织。将一些相对独立的代码变成函数,可以提高程序可读性与重用性,避免编写大量重复代码,简单理解就是将命令序列按格式写在一起,更方便重复使用命令序列

函数的定义

[ function ] 函数名(){

        参数获取 $1表示第一个参数 $2表示第二个参数以此类推

        命令序列

        [ return x ] //使用return或者exit可以显式的结束函数

}

函数的调用

函数名 [参数1] [参数2]

函数的作用范围

写在/etc/profile的函数可以直接在终端调用

在 Shell 脚本中函数的执行并不会开启一个新的子 Shell,而是仅在当前定义的 Shell 环境中有效。如果 Shell 脚本中的变量没有经过特殊设定,默认在整个脚本中都是有效的。在编写脚本时,有时需要将变量的值限定在函数内部,可以通过内置命令 local 来实现。函数内部变量的使用,可以避免函数内外同时出现同名变量对脚本结果的影响

函数变量作用范围示例
#脚本内容
#!/bin/bash
myfun () {
    local i
    i=8
    echo $i
}
i=9
myfun     调用函数
echo $i   

#执行脚本
[root@localhost~]# chmod +x fun_scope.sh
[root@localhost ~]# ./fun_scope.sh
8
9

函数的参数 

在使用函数参数时,函数名称在前参数在后,函数名和参数之间用空格分隔,可以有多个参数,参数使用$1、$2、$3……的方式表示。以此类推,从第 10 个参数开始,调用方法为${10},不加大括号无法调用成功,使用函数时函数只能使用$1、$2、$3……方式调用脚本传到函数的参数而不能使用这种方式调用脚本外传递到脚本的参数

递归函数

Shell 也可以实现递归函数,就是可以调用自己本身的函数。在 Linux 系统上编写 Shell 脚本的时候,经常需要递归遍历系统的目录和文件,列出目录下的文件和目录,逐层递归列出

#!/bin/bash

recursion() {
    dir="$1"    //传进来的目录参数
    for entry in ${dir}/* ;do    //遍历目录下的文件和目录
        if [ -d ${entry} ];then
            echo "目录:${entry}"
            recursion ${entry}    //函数自调用
        elif [ -f ${entry} ];then
            echo "文件:${entry}"
        fi
    done
}
if [ $# -ne 1 ];then
    echo "运行脚本时请传递一个参数:递归的目录"
    exit 1
fi
recursion $1

数组 

定义数组

  1. 方法一
    1. 数组名=(值 值 值 值 值 ...) //下标以0开始
  2. 方法二:
    1. 数组名=([0]=value[1]=value [2]=value ...)
  3. 方法三:
    • 列表名=”value0 value1 value2 ...”
    • 数组名=($列表名)
  4. 方法四:
    1. 数组名[0]=”value” 数组名[1]=”value” 数组名[2]=”value”

数组取值

  • 数组取值: ${数组名[数组下标]}
  • 数组全部值: ${数组名[*/@]} //数组全部的内容[]内书写*或者@
  • 数组长度:${#数组名[*/@]} //数组长度
  • 数组取值范围取值:${数组名[ @/*]:起始下标:长度}
  • 数组输出替换:${数组名[@/*]/匹配字符/替换字符}
  • unset 数组 /变量 //删除数组或者变量
  • 28
    点赞
  • 19
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值