while read line只能读一行内容

时间:2021-11-30 18:48:35

今天写了个脚本,主要功能是要远程登录一批机器执行命令。自然而然想到的是把机器ip列表放到文件中,然后使用while read line,在循环里用ssh远程登录并执行命令。脚本大概如下:

#!/bin/bash 
file=$1

while read line
do
    echo "ip:${line}===================="
    ssh root@${line} uname -a
done < ${file}

但是奇怪的是,我的file里有一批ip地址,但最终只执行了一次循环,也就是只对第一个ip地址进行了远程命令执行。我还一直以为是脚本或者系统配置有问题,但是把ssh命令注释,运行又是正常的,能读到所有ip。百思不得其解。

最后才发现这个和while语句的输入缓存有关。

执行while语句时,file文件的内容都已经读入缓冲区并重定向给了整个while语句。然后在while循环中调用read语句时,就会读取到下一行内容。然而因为ssh命令默认从标准输入里读取数据,就会把刚才读入while循环里缓冲区的内容使用掉。当下一次执行read时就无法获取输入,while循环也就结束了。

下面我们来实际验证一下:

[root@localhost /home]# cat file 
192.168.0.119
192.168.0.120
192.168.0.121
[root@localhost /home]# cat test.sh 
#!/bin/bash 
file=$1

while read line
do
    echo "ip:${line}===================="
    ssh root@${line} cat
done < ${file}
[root@localhost /home]# sh test.sh file 
ip:192.168.0.119====================
192.168.0.120
192.168.0.121
[root@localhost /home]# 

由上可见,在执行ssh命令时,已经把缓冲区的数据全部使用,也就导致while循环只能读取到一行的问题。

要怎么才能避免这种情况呢?
有两种办法:
1、指定ssh命令的输入为/dev/null
既然ssh默认读取标准输入,那我们就将其输入重定向即可。

ssh root@${line} cat < /dev/null

2、ssh命令使用-n参数,命令手册是这么解释-n参数的,

-n      Redirects stdin from /dev/null (actually, prevents reading from stdin).

因此,同样能达到我们的目的。

算是又涨见识了。。。。