跟散仙学shell编程(三)

散仙在上一篇文章里写了关于如何使用if-else以及case等流程控制的用法,本篇呢,来看下如何在shell里面使用更多的结构化命令,那么就是我们的循环了。


在shell里面基本的for循环的命令格式:
for var in list
do
commands
done
[search@h1 fortest]$ cat t1.sh 


for t in i have a dog
do
echo $t
done
[search@h1 fortest]$ sh t1.sh
i
have
a
dog
[search@h1 fortest]$


注意这个t变量,与其他编程语言不通,在for循环外面这个变量仍然有效:

[search@h1 fortest]$ cat t1.sh 


for t in i have a dog
do
echo $t
done


echo -n "t变量是否在循环外有效呢?我们来看看:"

echo $t

t="覆盖t"


echo $t


[search@h1 fortest]$ sh t1.sh
i
have
a
dog
t变量是否在循环外有效呢?我们来看看:dog
覆盖t
[search@h1 fortest]$

默认的shell分隔符是按空格分隔的,这一点需要注意,如果我们的字符串中本身含有空格,或其他特殊符号,有2中解决办法:
(1)用\转义类似单引号或双引号
(2)加上双引号表名是一个整体
[search@h1 fortest]$ cat t2.sh 


for var in "xi an" isn\'t a city
do

echo $var

done
[search@h1 fortest]$ sh t2.sh
xi an
isn't
a
city
[search@h1 fortest]$


从一个变量中读取数据:
[search@h1 fortest]$ cat a.sh 


list="a b c d e"
list=$list" f"


for var in $list

do
echo $var
done


echo "结束时间:"`date`
[search@h1 fortest]$ sh a.sh
a
b
c
d
e
f
结束时间:2014年 08月 12日 星期二 03:57:41 CST
[search@h1 fortest]$

从一个文本文件里读取值,并打印:
[search@h1 fortest]$ cat b.txt 
中国
日本
美国
法国

[search@h1 fortest]$ cat c.sh


for var in `cat b.txt`
do
echo $var
done

[search@h1 fortest]$ sh c.sh
中国
日本
美国
法国
[search@h1 fortest]$


更改字段分隔符,默认的环境变量IFS控制内部字段分隔符,默认情况下对空格,制表符,换行符生效:我们可以修改这个值,来临时满足我们的业务需要,目标达成后记得还原其原来的值:

IFS.OLD=$IFS
IFS=$'\N'
<!-- 处理流程,处理完毕后还原 -->
IFS=$IFS.OLD

[search@h1 fortest]$ cat b.txt 
中国 中国2 中国3
日本 日本1
美国 美国2 美国3
法国

[search@h1 fortest]$ sh c.sh
中国
中国2
中国3
日本
日本1
美国
美国2
美国3
法国
[search@h1 fortest]$ cat d.sh


#修改IFS环境变量分隔符,记得修改完,记得还原

echo "当前的IFS的值: "$IFS

IFSOLD=$IFS
IFS=$'\n'


for var in `cat b.txt`
do

echo $var

done


IFS=$IFSOLD


echo "最后还原IFS的值: "$IFS


[search@h1 fortest]$ sh d.sh
当前的IFS的值:
中国 中国2 中国3
日本 日本1
美国 美国2 美国3
法国
最后还原IFS的值:
[search@h1 fortest]$

学会使用IFS的使用,我们的解析方式就很灵活。
下面使用for循环,遍历目录

[search@h1 ~]$ cat d.sh 


for var in ./*

do

if [ -d "$var" ] ; then
echo "this is dir"$var
elif [ -f "$var" ] ; then
echo "this is file"$var
else
echo "i don't know it's type!"
fi

done


echo "end : "`date`
[search@h1 ~]$ sh d.sh
this is file./abc1.txt
this is file./abc.xml
this is file./apache-hive-0.13.1-bin.tar.gz
this is file./a.txt
this is file./b.txt
this is file./count.txt
this is file./c.txt
this is file./d.sh
this is dir./filehivebak
this is file./formathadoop.sh
this is dir./fortest
this is dir./hadoop
this is dir./hbase
this is dir./hbase-0.96.2-hadoop2
this is file./hbase-0.96.2-hadoop2-bin.tar.gz
this is dir./hive
this is dir./test
this is file./yarn-site.xml
end : 2014年 08月 12日 星期二 04:26:01 CST
[search@h1 ~]$


下面看类C语言的风格,注意双括号之间不需要空格:
[search@h1 fortest]$ cat e.sh 


for (( i=1; i<10; i++ ))
do

echo "数字: "$i

done
[search@h1 fortest]$ sh e.sh
数字: 1
数字: 2
数字: 3
数字: 4
数字: 5
数字: 6
数字: 7
数字: 8
数字: 9
[search@h1 fortest]$


使用多个变量:

[search@h1 fortest]$ cat aa.sh 


for (( a=1,b=10; a<=10 ; a++,b-- ))
do

echo "$a $b"

done
[search@h1 fortest]$ sh aa.sh
1 10
2 9
3 8
4 7
5 6
6 5
7 4
8 3
9 2
10 1
[search@h1 fortest]$


shell里面的while循环成立的条件是test命令返回的状态码为0的情况,如果非0则退出循环,这一点与其他的编程语言返回true,false不太一样:
[search@h1 fortest]$ cat bb.sh 

var=10
while [ $var -gt 0 ]
do
echo $var
var=$[ $var-1 ]

done

[search@h1 fortest]$ sh bb.sh
10
9
8
7
6
5
4
3
2
1
[search@h1 fortest]$


在shell里面还有另外一种循环util,它和while的工作方式相反,只有当状态码返回为0时,循环才会结束:

[search@h1 fortest]$ cat cc.sh 


var=100
until [ $var -eq 0 ]
do

echo $var
var=$[ $var - 25 ]
done

[search@h1 fortest]$ sh cc.sh
100
75
50
25
[search@h1 fortest]$



简单看下嵌套for循环:
[search@h1 fortest]$ cat ff.sh 


echo "外层循环一次,内层循环一遍! "

for (( i=1; i<10 ; i++ ))
do
echo "我是外层循环: "$i

for (( j=1; j<=3 ; j++ ))

do

echo " 我是内层循环: "$j


done


done


echo "结束: "`date`
[search@h1 fortest]$ sh ff.sh
外层循环一次,内层循环一遍!
我是外层循环: 1
我是内层循环: 1
我是内层循环: 2
我是内层循环: 3
我是外层循环: 2
我是内层循环: 1
我是内层循环: 2
我是内层循环: 3
我是外层循环: 3
我是内层循环: 1
我是内层循环: 2
我是内层循环: 3
我是外层循环: 4
我是内层循环: 1
我是内层循环: 2
我是内层循环: 3
我是外层循环: 5
我是内层循环: 1
我是内层循环: 2
我是内层循环: 3
我是外层循环: 6
我是内层循环: 1
我是内层循环: 2
我是内层循环: 3
我是外层循环: 7
我是内层循环: 1
我是内层循环: 2
我是内层循环: 3
我是外层循环: 8
我是内层循环: 1
我是内层循环: 2
我是内层循环: 3
我是外层循环: 9
我是内层循环: 1
我是内层循环: 2
我是内层循环: 3
结束: 2014年 08月 12日 星期二 04:50:47 CST
[search@h1 fortest]$


下面来看一个小李子实战:
[search@h1 fortest]$ cat hh.sh 


IFSOLD=$IFS
IFS=$'\n'


for v in `cat /etc/passwd`
do
echo "每一行的值 $v"
# echo "ifs1: $IFS"
IFS=:
# echo "ifs2: $IFS"

for v2 in $v

do
echo " 按冒号拆分的值:$v2 "
done


#IFS=$'\n'


done


IFS=$IFSOLD


[search@h1 fortest]$ sh hh.sh
每一行的值 root:x:0:0:root:/root:/bin/bash
按冒号拆分的值:root
按冒号拆分的值:x
按冒号拆分的值:0
按冒号拆分的值:0
按冒号拆分的值:root
按冒号拆分的值:/root
按冒号拆分的值:/bin/bash
每一行的值 bin:x:1:1:bin:/bin:/sbin/nologin
按冒号拆分的值:bin
按冒号拆分的值:x
按冒号拆分的值:1
按冒号拆分的值:1
按冒号拆分的值:bin
按冒号拆分的值:/bin
按冒号拆分的值:/sbin/nologin
每一行的值 daemon:x:2:2:daemon:/sbin:/sbin/nologin
按冒号拆分的值:daemon
按冒号拆分的值:x
按冒号拆分的值:2
按冒号拆分的值:2
按冒号拆分的值:daemon
按冒号拆分的值:/sbin
按冒号拆分的值:/sbin/nologin
每一行的值 adm:x:3:4:adm:/var/adm:/sbin/nologin
按冒号拆分的值:adm
按冒号拆分的值:x
按冒号拆分的值:3
按冒号拆分的值:4
按冒号拆分的值:adm
按冒号拆分的值:/var/adm
按冒号拆分的值:/sbin/nologin
每一行的值 lp:x:4:7:lp:/var/spool/lpd:/sbin/nologin
按冒号拆分的值:lp
按冒号拆分的值:x
按冒号拆分的值:4
按冒号拆分的值:7
按冒号拆分的值:lp
按冒号拆分的值:/var/spool/lpd
按冒号拆分的值:/sbin/nologin
每一行的值 sync:x:5:0:sync:/sbin:/bin/sync
按冒号拆分的值:sync
按冒号拆分的值:x
按冒号拆分的值:5
按冒号拆分的值:0
按冒号拆分的值:sync
按冒号拆分的值:/sbin
按冒号拆分的值:/bin/sync
每一行的值 shutdown:x:6:0:shutdown:/sbin:/sbin/shutdown
按冒号拆分的值:shutdown
按冒号拆分的值:x
按冒号拆分的值:6
按冒号拆分的值:0
按冒号拆分的值:shutdown
按冒号拆分的值:/sbin
按冒号拆分的值:/sbin/shutdown
每一行的值 halt:x:7:0:halt:/sbin:/sbin/halt
按冒号拆分的值:halt
按冒号拆分的值:x
按冒号拆分的值:7
按冒号拆分的值:0
按冒号拆分的值:halt
按冒号拆分的值:/sbin
按冒号拆分的值:/sbin/halt
每一行的值 mail:x:8:12:mail:/var/spool/mail:/sbin/nologin
按冒号拆分的值:mail
按冒号拆分的值:x
按冒号拆分的值:8
按冒号拆分的值:12
按冒号拆分的值:mail
按冒号拆分的值:/var/spool/mail
按冒号拆分的值:/sbin/nologin
每一行的值 uucp:x:10:14:uucp:/var/spool/uucp:/sbin/nologin
按冒号拆分的值:uucp
按冒号拆分的值:x
按冒号拆分的值:10
按冒号拆分的值:14
按冒号拆分的值:uucp
按冒号拆分的值:/var/spool/uucp
按冒号拆分的值:/sbin/nologin
每一行的值 operator:x:11:0:operator:/root:/sbin/nologin
按冒号拆分的值:operator
按冒号拆分的值:x
按冒号拆分的值:11
按冒号拆分的值:0
按冒号拆分的值:operator
按冒号拆分的值:/root
按冒号拆分的值:/sbin/nologin
每一行的值 games:x:12:100:games:/usr/games:/sbin/nologin
按冒号拆分的值:games
按冒号拆分的值:x
按冒号拆分的值:12
按冒号拆分的值:100
按冒号拆分的值:games
按冒号拆分的值:/usr/games
按冒号拆分的值:/sbin/nologin
每一行的值 gopher:x:13:30:gopher:/var/gopher:/sbin/nologin
按冒号拆分的值:gopher
按冒号拆分的值:x
按冒号拆分的值:13
按冒号拆分的值:30
按冒号拆分的值:gopher
按冒号拆分的值:/var/gopher
按冒号拆分的值:/sbin/nologin
每一行的值 ftp:x:14:50:FTP User:/var/ftp:/sbin/nologin
按冒号拆分的值:ftp
按冒号拆分的值:x
按冒号拆分的值:14
按冒号拆分的值:50
按冒号拆分的值:FTP User
按冒号拆分的值:/var/ftp
按冒号拆分的值:/sbin/nologin
每一行的值 nobody:x:99:99:Nobody:/:/sbin/nologin
按冒号拆分的值:nobody
按冒号拆分的值:x
按冒号拆分的值:99
按冒号拆分的值:99
按冒号拆分的值:Nobody
按冒号拆分的值:/
按冒号拆分的值:/sbin/nologin
每一行的值 vcsa:x:69:69:virtual console memory owner:/dev:/sbin/nologin
按冒号拆分的值:vcsa
按冒号拆分的值:x
按冒号拆分的值:69
按冒号拆分的值:69
按冒号拆分的值:virtual console memory owner
按冒号拆分的值:/dev
按冒号拆分的值:/sbin/nologin
每一行的值 saslauth:x:499:76:"Saslauthd user":/var/empty/saslauth:/sbin/nologin
按冒号拆分的值:saslauth
按冒号拆分的值:x
按冒号拆分的值:499
按冒号拆分的值:76
按冒号拆分的值:"Saslauthd user"
按冒号拆分的值:/var/empty/saslauth
按冒号拆分的值:/sbin/nologin
每一行的值 postfix:x:89:89::/var/spool/postfix:/sbin/nologin
按冒号拆分的值:postfix
按冒号拆分的值:x
按冒号拆分的值:89
按冒号拆分的值:89
按冒号拆分的值:
按冒号拆分的值:/var/spool/postfix
按冒号拆分的值:/sbin/nologin
每一行的值 sshd:x:74:74:Privilege-separated SSH:/var/empty/sshd:/sbin/nologin
按冒号拆分的值:sshd
按冒号拆分的值:x
按冒号拆分的值:74
按冒号拆分的值:74
按冒号拆分的值:Privilege-separated SSH
按冒号拆分的值:/var/empty/sshd
按冒号拆分的值:/sbin/nologin
每一行的值 mysql:x:27:27:MySQL Server:/var/lib/mysql:/bin/bash
按冒号拆分的值:mysql
按冒号拆分的值:x
按冒号拆分的值:27
按冒号拆分的值:27
按冒号拆分的值:MySQL Server
按冒号拆分的值:/var/lib/mysql
按冒号拆分的值:/bin/bash
每一行的值 search:x:500:500:search,hah,132532,111:/home/search:/bin/bash
按冒号拆分的值:search
按冒号拆分的值:x
按冒号拆分的值:500
按冒号拆分的值:500
按冒号拆分的值:search,hah,132532,111
按冒号拆分的值:/home/search
按冒号拆分的值:/bin/bash
[search@h1 fortest]$


除此之外,我们还可以在shell里面使用continue,和break关键字:

[search@h1 fortest]$ cat gg.sh 


for (( i=1; i<10 ; i++ ))

do
if (( $i % 2 == 0 )) ;then
echo "双数: $i"
else
echo "单数: $i"
fi



done
[search@h1 fortest]$ sh gg.sh
单数: 1
双数: 2
单数: 3
双数: 4
单数: 5
双数: 6
单数: 7
双数: 8
单数: 9
[search@h1 fortest]$


break n n代表结束循环的层数
continue n n代表跳过当前循环的层数
这一点与我们的JAVA不太一样,break用法例子:
[search@h1 fortest]$ cat gg.sh 


for (( i=1; i<10 ; i++ ))

do
if (( $i == 7 )) ;then
echo "结束循环"
break;

else
echo "单数: $i"
fi



done
[search@h1 fortest]$ sh gg.sh
单数: 1
单数: 2
单数: 3
单数: 4
单数: 5
单数: 6
结束循环
[search@h1 fortest]$


下面是continue的用法:
[search@h1 fortest]$ cat gg.sh 


for (( i=1; i<10 ; i++ ))

do
if (( $i == 7 )) ;then
echo "跳过当前循环的值"

continue;
else
echo "数: $i"
fi



done
[search@h1 fortest]$ sh gg.sh
数: 1
数: 2
数: 3
数: 4
数: 5
数: 6
跳过当前循环的值
数: 8
数: 9
[search@h1 fortest]$


最后我们再看下,循环的结果如何重定向到另外一个命令:

[search@h1 fortest]$ cat hx.sh 


echo "使用for循环的结果直接重定向并排序: "
for var in Brand Abstract China Zokeeber Zabbix Ganglia

do
echo "排序后的值是:"$var

done | sort


[search@h1 fortest]$ sh hx.sh
使用for循环的结果直接重定向并排序:
排序后的值是:Abstract
排序后的值是:Brand
排序后的值是:China
排序后的值是:Ganglia
排序后的值是:Zabbix
排序后的值是:Zokeeber
[search@h1 fortest]$


下面是结果排序,再写入一个文件里:
[search@h1 fortest]$ cat hx.sh 


echo "使用for循环的结果直接重定向写入一个文件中 "
for var in Brand Abstract China Zokeeber Zabbix Ganglia

do
echo "值是:"$var

done | sort >> result.txt


[search@h1 fortest]$ sh hx.sh
使用for循环的结果直接重定向写入一个文件中
[search@h1 fortest]$ cat result.txt
值是:Abstract
值是:Brand
值是:China
值是:Ganglia
值是:Zabbix
值是:Zokeeber
[search@h1 fortest]$


通过管道命令重定向,我们可以更加容易操作我们的程序!
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值