我用过bash shell,而且时间不短了。但我从来没学过shell,至少没有像C++这么认真去学。平时写些基本的脚本没问题,不懂也可以google、百度。可在2014最后一天,掉坑里了。

  其实脚本也很简单,代码基本如下:

#!/bin/bash

declare -A FILES    #关联数组 bash version >= 4.0

cat EX_FILES | while read line
do
    key=`echo $line | awk '{print $1}'`
    value=`echo $line | awk '{print $2}'`
    FILES[$key]=$value
done

echo ${FILES[@]}

其实也就是从文件每次读出一行,再从一行中按列分解出key、value,然后再赋值到map里去。可以,输出的map居然为空。于是,开始了几个钟头的折腾。

  一开始以为是关联数组的原因,因为是第一次用关联数组。可是google、百度,man bash,都是这么用的。在FILES[$key]=$value这里打印也是正确的,到while之外FILES却是空的。然后怀疑是因为key是字符串的原因,于是另起一个脚本测试,key为字符串完全没问题。想想这种情况应该是FILES成了局部变量(按C++的方式,这么想的),可我声明FILES时没用local啊。难道是while read的原因?于是搜索了一下"while read 局部变量",搜索到了大量的sub shell相关的问题,也正是我的问题。

  在 cat EX_FILES | while ...done 这代码中,cat EX_FILES是在当前shell中运行,while ...done是在子shell中运行,所以当前shell的数组没有被赋值。解决的方法改一下就好:

#!/bin/bash

declare -A FILES
while read line
do
    key=`echo $line | awk '{print $1}'`
    value=`echo $line | awk '{print $2}'`
    FILES[$key]=$value
done < EX_FILES

echo ${FILES[@]}

这样要注意EX_FILES可能因为换行符的原因读取不了最后一行

posted on 2015-01-01 11:35  coding my life  阅读(408)  评论(0编辑  收藏  举报