今天写了个脚本,主要功能是要远程登录一批机器执行命令。自然而然想到的是把机器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).
因此,同样能达到我们的目的。
算是又涨见识了。。。。