前段时间和几位同事讨论过一个问题:Shell脚本里面怎样把一个数组传递到awk内部进行处理?
当时没有找到方法。前两天在QQ群里讨论awk的时候,无意间又聊起这个话题。机缘巧合之下找到一个思路,特此分享。
测试环境:
[root]# head -1 /etc/redhat-release
Red Hat Enterprise Linux Server release 6.5 (Santiago)
[root]# awk --version | head -1
GNU Awk 3.1.7
|
众所周知,Shell脚本里面把一个普通变量传递给awk是非常简单的,直接用 -v 参数赋值就行了。
-
str1="Hello World"
-
awk -v str2="$str1" 'BEGIN{print str2}'
但是,要把一个数组传递给awk就不是那么简单的事情了。请看以下三个试验:
1. 简单的数组可以先赋值后split
-
arr1=(A B C)
-
awk -v arr2="${arr1[*]}" 'BEGIN{split(arr2,arr3," "); print arr3[2]}'
2. 有些情况下很难找到合适的分隔符来进行split,因为某个数组元素可能会包含你想用来作为分隔符的那个字符,这时split之后无法得到希望的结果。所以这个方法不够严谨,尤其是当我们无法预测数组元素可能包含哪些字符的时候。
-
arr1=(A "B C" D)
-
awk -v arr2="${arr1[*]}" 'BEGIN{split(arr2,arr3," "); print arr3[2]}'
3. 可以借助export命令和awk的ENVIRON默认数组来实现这个功能
-
arr1=(A "B C" D)
-
for((i=0;i<${#arr1[*]};i++)); do
-
export arr1_m$i="${arr1[$i]}"
-
done
-
awk 'BEGIN{for(i in ENVIRON)if(i~/arr1_m/)print i "=" ENVIRON[i]}'
我这里只是为了演示功能,所以没有把export变量名的定义和awk内部的字符串匹配写的特别考究,大家可以根据实际情况进行调整(比如添加更多的限制条件等)。
结论:Shell脚本里面把一个数组传递到awk内部进行处理,技术上可行,但不建议在生产环境上使用。
不是有那么句名言嘛:
Always code as if the guy who ends up maintaining your code will be a violent psychopath who knows where you live.
阅读(1467) | 评论(0) | 转发(0) |