时间:2021-07-18 08:27:00 | 栏目:Shell | 点击:次
最近实现了一个shell脚本,功能简单来说就是从文件中按行读取然后将所有行拼接成一行写入一个文件,关键代码如下
path_all="" cat $1 | while read line do if [ "$path_all" == "" ];then path_all=$line else path_all=$path_all,$line fi done echo $path_all > $one_file
结果执行结果竟然是空!按照常理shell中的变量默认是全局变量,不会存在变量作用域的问题,于是上网查了下,发现问题出在while上
while循环读取文件中内容有两种写法,一种是管道符,一种是重定向,写法如下
管道符:
cat $file_name | while read line do #deal with line done
重定向
while read line do #deal with line done < $file_name
这两种做法的区别在于,重定向是内建命令,而管道符是非内建命令,
之所以我写的脚本出现了输出是空的问题,原因就在这里
linux执行shell时,会创建“子shell”运行shell中的命令,当运行到非内建指令时,会创建“孙shell”运行非内建指令
变量的作用于在每个shell中有效,所以,非内建指令中定义的这些变量就只能在孙shell运行,而在子shell中不生效,所以,即便我在while中给path_all赋值了,子shell中也不会获取到这个值。
解决这个问题的办法有两种,如下
如果不是必须使用管道符的方式写while循环,可以用重定向的写法,这种写法循环内的变量在子shell中是生效的,比较简便
如果非要使用管道符的方式,可以创建临时文件,用于存放孙shell中的输出
cat $file_name | while read line do echo $line >> $tmp_file done cat $tmp_file >$one_file
虽然两种方法都可以解决这个问题,但还是推荐第一种解决方法