awk, как изменить значение файла с помощью наборов параметров?

awk, как изменить значение файла с помощью наборов параметров?

Я пытаюсь сбросить свои данные с помощью awk. У меня есть данные из 54 столбцов, где я хочу изменить шесть значений столбцов с помощью следующего скрипта.

#!/bin/bash -x
# optimal parameters
set BI2=0.02841513
set DS2=0.37837487
set WS2=0.85392380
set D22=1.00048351
set D32=0.63893676
set DSMAX2=28.15795517

# change the calibration parameter
awk '{if($1==2) {
  $6 = "${BI2}"
  $7 = "${DS2}"
  $8 = "${DSMAX2}"
  $9 = "${WS2}"
  $25 = "${D22}"
  $26 = "${D32}"}
  print $0;}' soil_params > tmp
mv -f tmp soil_params

мои данные выглядят так:

999 0   123 65.183534   -147.655522 0.2 0.001   10  0.9 2   10.84   9.79    9.76    812.42  1546.28 1334.35 -999    -999    -999    18.579  68.139  26.914  548 0.1 0.3 0.192752    -0.8725 4   35.39   30.13   19.34   0.323   0.347   0.382   1160.4  1196.2  1208.9  2186.08 2253.52 2411.44 -9.8    0.582   0.489   0.443   0.396   0.4841  0.28    0.001   0.0005  383.7   0.1 0.1 0.1 1   16.733299   16.733299                                           
2   1   124 65.183534   -147.639897 ${BI2}  ${DS2}  ${DSMAX2}   ${WS2}  2   10.84   9.79    9.77    812.51  1544.96 1331.78 -999    -999    -999    18.579  68.277  26.915  676 0.1 ${D22}  ${D32}  -0.529166   4   35.45   30.19   19.37   0.323   0.347   0.382   1160.3  1196.3  1208.8  2185.87 2253.63 2411.11 -9.8    0.582   0.49    0.443   0.396   0.4851  0.28    0.001   0.0005  400.8   0.1 0.1 0.1 1   16.32   16.32                                                       

Когда я запускаю приведенный выше скрипт, он не меняет значение из назначенного набора параметров, а просто изменяет параметры, как вы можете

Не могли бы вы мне помочь в этом вопросе?

решение1

проблема, с которой вы столкнулись, заключается в том, что вы пытаетесь использовать переменные BASH внутри команды awk, которая защищена одинарными кавычками ( '...'). внутри одинарных кавычек bash будетнетразверните переменные (например ${BI2}, ), так что awk буквально увидит $6 = "${BI2}"и установит значение 6-го столбца в литерал "${B12}".

к счастью, вы можете легко создавать awkпеременные из shellпеременных, используя аргумент --assign(он же -v):

awk \
   -v BI2="${BI2}" \
   -v DS2="${DS2}" \
   -v DSMAX2="${DSMAX2}" \
   -v WS2="${WS2}" \
   -v D22="${D22}" \
   -v D32="${D32}" \
  '{if($1==2)
     {
      $6 = BI2; $7 = DS2; $8 = DSMAX2; $9 = WS2; $25 =D22; $26 =D32; 
     }
    print $0;
  }'

решение2

Вы заключаете awkскрипт в одинарные кавычки, это обычно рекомендуется, поскольку это не позволяет оболочке пытаться подставлять переменные, определенные awk, например $1. К сожалению, это также означает, что оболочка не может подставлять переменные, о которых знает только она, например $BI2. Если вместо этого вы заключите свой awkвызов в двойные кавычки:

awk "{if(\$1==2) {
  \$6 = ${BI2}
  \$7 = ${DS2}
  \$8 = ${DSMAX2}
  \$9 = ${WS2}
  \$25 = ${D22}
  \$26 = ${D32}}
  print \$0;}" soil_params > tmp

то оболочка заменит ${BI2}и т. д. Однако теперь переменные, которые awkиспользуют, например $1, должны быть записаны как , \$1чтобы предотвратить попытки оболочки предоставить им значения.

Связанный контент