在Linux / UNIX操作系统下,如何使用bash for loop重复执行某些任务?
如何使用for语句设置无限循环?
如何使用三参数进行循环控制表达式?
“ for循环”是bash编程语言的语句,它允许重复执行代码。 for循环被归类为迭代语句,即bash脚本中进程的重复。
例如,您可以运行UNIX命令或任务5次,或使用for循环读取和处理文件列表。 可以在shell提示符下或在shell脚本本身内使用for循环。
for循环语法
数字范围的语法如下:
for VARIABLE in 1 2 3 4 5 .. N
do
command1
command2
commandN
done
或
for VARIABLE in file1 file2 file3
do
command1 on $VARIABLE
command2
commandN
done
或
for OUTPUT in $(Linux-Or-Unix-Command-Here)
do
command1 on $OUTPUT
command2 on $OUTPUT
commandN
done
实例
这种for循环的特征是计数。范围由开始(#1)和结束数字(#5)指定。for循环为项列表中的每个成员执行一系列命令。下面是BASH中的一个典型示例,它使用for循环显示欢迎消息5次
#!/bin/bash
for i in 1 2 3 4 5
do
echo "Welcome $i times"
done
有时您可能需要设置一个step值(例如允许一个按两个s计数或向后计数)。最新的bash 3.0+版本内置了对设置范围的支持
#!/bin/bash
for i in {1..5}
do
echo "Welcome $i times"
done
Bash v4.0+内置了对使用{START..END..增量}语法
#!/bin/bash
echo "Bash version ${BASH_VERSION}..."
for i in {0..10..2}
do
echo "Welcome $i times"
done
输出结果:
Bash version 4.0.33(0)-release...
Welcome 0 times
Welcome 2 times
Welcome 4 times
Welcome 6 times
Welcome 8 times
Welcome 10 times
seq命令(过时的)
警告!seq命令打印一个数字序列,由于历史原因,它出现在这里。以下示例仅适用于较旧的bash版本。建议所有用户(bash v3.x+)使用上述语法。
seq命令可按如下方式使用。seq中的一个典型例子如下
#!/bin/bash
for i in $(seq 1 2 20)
do
echo "Welcome $i times"
done
没有很好的理由使用诸如seq之类的外部命令来计数和递增for循环中的数字,因此建议您避免使用seq。内置命令是快速的。
三元表达式bash for循环语法
这种for循环与C编程语言有一个共同的传统。其特征是一个三参数循环控制表达式;由一个初始化器(EXP1)、一个循环测试或条件(EXP2)和一个计数表达式(EXP3)组成。
for (( EXP1; EXP2; EXP3 ))
do
command1
command2
command3
done
下面是bash中的一个典型的三元表达式示例
#!/bin/bash
for (( c=1; c<=5; c++ ))
do
echo "Welcome $c times"
done
输出结果:
Welcome 1 times
Welcome 2 times
Welcome 3 times
Welcome 4 times
Welcome 5 times
如何使用for作为无限循环?
可以使用空表达式创建Infinite for循环,比如
#!/bin/bash
for (( ; ; ))
do
echo "infinite loops [ hit CTRL+C to stop]"
done
带断点的条件退出
您可以在for循环中使用break语句提前退出。您可以使用break从FOR、WHILE或UNTIL循环中退出。for循环中的General break语句
for I in 1 2 3 4 5
do
statements1 #Executed for all values of ''I'', up to a disaster-condition if any.
statements2
if (disaster-condition)
then
break #Abandon the loop.
fi
statements3 #While good and, no disaster-condition.
done
以下shell脚本将通过所有存储在/ etc目录中的文件。 找到/etc/resolv.conf文件时,将跳出for循环。
#!/bin/bash
for file in /etc/*
do
if [ "${file}" == "/etc/resolv.conf" ]
then
countNameservers=$(grep -c nameserver /etc/resolv.conf)
echo "Total ${countNameservers} nameservers defined in ${file}"
break
fi
done
使用continue语句
若要继续封闭FOR、WHILE或UNTIL循环的下一个迭代,请使用continue语句。
for I in 1 2 3 4 5
do
statements1 #Executed for all values of ''I'', up to a disaster-condition if any.
statements2
if (condition)
then
continue #Go to next iteration of I in the loop and skip statements3
fi
statements3
done
此脚本备份命令行中指定的所有文件名。如果.bak文件存在,它将跳过cp命令。
#!/bin/bash
FILES="$@"
for f in $FILES
do
# if .bak backup file exists, read next file
if [ -f ${f}.bak ]
then
echo "Skiping $f file..."
continue # read next file and skip the cp command
fi
# we are here means no backup file exists, just use cp command to copy file
/bin/cp $f $f.bak
done
放在一起
Bash for循环对于自动化IT中的重复任务非常有用。让我们看看如何在多个Linux或Unix服务器上运行一个简单的命令(例如正常运行时间)
for s in server1 server2 server3
do
ssh vivek@${s} "uptime"
done
或者将echo命令与命令替换合并,如下所示
for s in server1 server2 server3
do
echo "Server ${s}: $(ssh vivek@${s} uptime)"
done
输出结果
Server server1: 09:34:46 up 12 days, 21:57, 0 users, load average: 0.08, 0.09, 0.09
Server server2: 09:34:50 up 17 days, 2:30, 0 users, load average: 0.03, 0.03, 0.00
Server server3: 09:34:53 up 17 days, 2:31, 0 users, load average: 0.04, 0.04, 0.00
在此标准bash for loop示例中,如果我们有基于Debian / Ubuntu的服务器,我们将使用yum命令或apt命令/ apt-get命令更新所有基于CentOS / RHEL的服务器:
## CENTOS/RHEL example (for fedora replace yum with dnf) ##
for s in server0{1..8}
do
echo "*** Patching and updating ${s} ***"
ssh root@${s} -- "yum -y update"
done
这是简单但有用的shell脚本示例
#!/usr/bin/env bash
# Purpose: Update all my Linode servers powered by Debian/Ubuntu Linux
# Author: Vivek Gite under GPL v2.x+
# ----------------------------------------
log="/tmp/apt-get.log"
>"${log}"
for s in ln.cbz0{1..5}
do
echo "Updating and patching $s, please wait..." | tee -a "${log}"
ssh root@${s} -- apt-get -q -y update >/dev/null
ssh root@${s} -- DEBIAN_FRONTEND=noninteractive apt-get -y -q upgrade >>"${log}"
done
echo "Check $log file for details."
了解为什么我们使用DEBIAN_FRONTEND apt-get变量来避免更新期间出现任何提示。 最好是出于自动化目的设置ssh密钥或从Linux / Unix cron作业运行脚本。
总结
您通过各种示例学习了如何使用bash for loop。 For循环可以节省时间,并可以帮助您自动完成微小的任务。 但是,对于复杂的IT自动化任务,您应该使用Ansible,Salt,Chef,pssh等工具。