0

所以我想根据第 8 列将一个相当大的文件分成几个小文件。所以我写了这个脚本:

#!/bin/bash
run_command(){
eval ${1}
wait
}
chInput=("1" "2" "3" "4" "5" "6" "7" "8" "9" "10" "11" "12" "13" "14" "15" "16" "17" "18" "19" "Z" "T" "G" "F" "A" "D" "P")
sampInput=("heyA")

for ((x=0;x<${#chInput[@]};x++));do
com="awk -F'\t' '$8=="${chInput[x]}"' /home/location/"$sampInput"_This_P.txt > "$sampInput"Ch"${chInput[x]}".txt"
run_command "${com}"
done

但它不起作用,因为

'$8=="

awk: ==1
awk: ^ syntax error
awk: ==2
awk: ^ syntax error
awk: ==3
awk: ^ syntax error
awk: ==4
awk: ^ syntax error

但只是做

awk -F'\t' '$8==1' /home/location/heyA_This_P.txt > Ch1.txt

从命令行确实有效

我能做些什么来解决这个问题?

4

1 回答 1

2

严重的问题是双引号;$8在您分配变量时,将被某些东西(可能根本没有)替换。您可以尝试使用带有适当转义的单引号,但真正的解决方案可能是深呼吸并重新开始,而无需eval在变量中包含任何一个或 Awk 脚本。

无论如何,这种椒盐脆饼逻辑的目的是什么?您可能应该阅读并牢记 http://mywiki.wooledge.org/BashFAQ/050中的建议

这是解决您的问题的快速尝试:

#!/bin/bash

chInput=("1" "2" "3" "4" "5" "6" "7" "8" "9" "10" "11" "12" "13" "14" "15" "16" "17" "18" "19" "Z" "T" "G" "F" "A" "D" "P")
sampInput=("heyA")

for ((x=0;x<${#chInput[@]};x++));do
    awk -F'\t' '$8=="'"${chInput[x]}"'"' /home/location/"$sampInput"_This_P.txt > "$sampInput"Ch"${chInput[x]}".txt
done

特别注意用于插入"${chInput[X]}"脚本的构造(实际上,除了删除变量和其他内容之外,这实际上是我唯一更改的eval内容)。那是单引号中的字符串,与双引号中的字符串相邻,与单引号中的字符串相邻,在 Bash 中计算为单个字符串。所以'foo'"bar"'baz'评估为foobarbaz并且类似地'"foo"'相邻"'bar'"评估为"foo"'bar'。在这里,'$8=="'与相邻的"${chInput[x]}"相邻'"'计算$8=="..."在分配时替换双引号中的内容的位置。

(你也不需要数组;你可以这样做

for c in "1" "2" "3" "4" "5" "6" "7" "8" "9" "10" "11" "12" \
         "13" "14" "15" "16" "17" "18" "19" "Z" "T" "G" "F" \
         "A" "D" "P"
do
    awk -F'\t' '$8=="'"$c"'"' /home/location/"$sampInput"_This_P.txt > "${sampInput}Ch$c.txt"
done

并与 Classic Bourne shell 兼容。)

于 2013-04-12T20:19:08.383 回答