有这样一个需求:
• 循环遍历文件每一行:流程控制语句
– 定义一个计数器num
– 打印num正好是文件行数
目前有四种方法可以解决:
1.for循环遍历
oldIFS=$IFS
IFS=$'\n'
num=0
for i in `cat data.txt` ;do
echo $i
((num++))
done
echo num:$num
IFS=$oldIFS
因为切割文件时,空格和回车换行符都会切割,而我们只想要切割回车换行符,所以应该改下IFS,让它只根据回车换行符切割这样每次得到的是完整的一行数据。将它原本的设置赋值给一个变量,循环完毕恢复。
遍历文件全部内容用cat并用``将命令的结果作为循环体,每打印一行数据,行数加1,循环完毕,打印num;
2.for循环遍历
num=0
lineNum=`cat data.txt | wc -l`
for((i=1;i<=lineNum ;i++));do
head -$i data.txt | tail -1
((num++))
done
echo num:$num
获取文件的总行数,利用管道和命令替换,用wc获取全部文件内容的总行数并赋值给一个变量lineNum,
遍历,
利用head获取前i行并利用管道查看最后一行,最终查看的就是第i行也就是当前行内容,累加,循环完毕打印num
3.while循环
exec 8<& 0
exec 0< data.txt # instream in = new instream(data.txt)
num=0
while read line ;do
echo $line
((num++))
done
echo num:$num
exec 0<& 8
exec 8<& -
核心思想是将输入改为文件一行一行的读。
定义一个操作符8指向0标准输入,为的是循环结束后恢复0的标准输入。
while循环 ,read以换行符为切点一行一行的读。
循环结束,打印num,恢复0的输入为标准输入
4.while循环
简便写法:
num=0
while read line ;do
echo $line
((num++))
done 0< data.txt
echo num:$num
注意:!!!
export num=0
cat data.txt | while read line;do
echo $line
((num++))
echo $num
done
echo num:$num
这种写法会造成num不对。因为管道命令会为左右开辟两个子进程,右边子进程结束子变量num销毁,并不会对主进程变量num产生影响,此时打印主进程num将为初始值0.