在Linux的流程控制部分有一个小案例可以将流程控制、管道、重定向、命令替换的知识整合到一起练习,现在就分享一下这个案例。
循环遍历文件每一行,定义一个计数器num,要求打印num正好是文件行数。
这个案例的几种写法如下:
增强for循环的方式:
在这种方法使用之前先要补充一个知识点:
IFS 是一种 set 变量,当 shell 处理"命令替换"和"参数替换"时,shell 根据 IFS 的值,默认是 space, tab, newline 来拆解读入的变量,然后对特殊字符进行处理,最后重新组合赋值给该变量。
查看变量 IFS 的值。
直接输出IFS是看不到的,把它转化为二进制就可以看到了,"040"是空格,“011"是Tab,“012"是换行符”\n” 。最后一个 012 是因为 echo 默认是会换行的。
oldIFS=$IFS //改变词的切分,设置为以换行符切分,那么得到的就是一整行
IFS=$'\n'
num=0 //起初定义一个变量
for i in `cat data.txt` ;do //将文件中的内容全部读出来,需要命令替换
echo $i //每循环一次打印
((num++)) //打印完num+1
done
echo num = $num //最后输出结果
IFS=$oldIFS //再将原来的IFS的值重新赋值回来
基于步进for循环的方式:
num=0
lineNum=`cat data.txt | wc -l` //给变量赋值,用cat命令得到文件的内容,用wc统计总共有多少行
for ((i=1;i<=lineNum;i++));do //至少循环文件的行数
head -$i data.txt | tail -1 //循环第一次打印文件的第一行,循环第二次打印文件的第二行,循环第三次打印文件的第三行;
//用head和tail的组合,循环第二次得到前两行,但是只取最后一行即得到文件的第二行
((num++))
done
echo num = $num
while循环的第一种写法:
知识点铺垫:
read是shell builtin的
while是shell的keyword关键字
exec 8<& 0 //准备一个文件描述符(此处是8),这个文件描述符的来源是0,来自键盘
exec 0< data.txt //然后让0来自于data.txt,只要有程序想读0那么就会读到该文件的内容,
//约等于Java的inputstream is = new inputstream(data.txt)。也就是说如果这个脚本运行了变成一个进程,
//这句话是修改脚本子进程的标准输入,让其能够从文件中读到内容
num=0
while read line;do //read读会开启键盘输入,但是在这之前已经将输入指向了一个数据文件,因此每
echo $line
((num++))
done
exec 0<& 8
exec 8<& -
echo num = $num
以上写法的精简版:
num=0
while read line;do
echo $line
((num++))
done 0< data.txt //while每次循环不需要重新再绑定0
echo num = $num
避免踩坑:这种写法只会不断输出第一行的数据,进入死循环
num=0
while read line 0< data.txt;do //每次while循环的时候都重新让0来自文件
echo $line
((num++))
done
echo num = $num
可能有人还想到用管道标准输出和标准输入,但是使用管道时要很小心,避免踩坑。例如:
num=0
cat data.txt | while read line ;do //前一个命令的输出到后一个命令的输入,管道左边是一个子进程,右边也是一个子进程,
//结果输出的num值还是0;有可能想到一个解决办法,在num之前加上export,这样做也是不对的,
//因为环境变量是导出而非共享,子进程的变化父进程得不到
echo $line
((num++))
done
echo num = $num