¿Awk no imprime $2 de un archivo?

¿Awk no imprime $2 de un archivo?

Estoy ejecutando este script de shell

 #!/bin/sh
 for line in `cat file.txt`
 do
 Am=`echo $line| awk '{print $1}'`
 Bm=`echo $line| awk '{print $2}'`
 #my command Am +Bm.......
 done

Me sale esto para la primera línea:

  + cat file.txt
  + echo A
  + awk {print $1}
  + AM=A
  + echo A
  + awk {print $2}
  + Bm=

como puedes ver no recoge el valor de Bm

     $ cat file.txt
     A 19
     B 81
     C 76
     D 31

   $ awk '{print $2}' file.txt
    19
    81
    76
    31

  $ awk '{print $1}' file.txt
    A
    B
    C
    D

Respuesta1

for line in $(cat file.txt)iterará sobre cada pegote del archivo, uno por uno. La entrada enviada a awksolo tendrá una sola columna de entrada, por lo que $2siempre será nula.

Respuesta2

IFSdivide de forma predeterminada el carácter de espacio para que su forbucle realmente recorra cada "palabra"

$ for x in `echo a 19 b whatever`; do echo $x; done
a
19
b
whatever
$ 

Un mejor enfoque podría ser eliminar tanto el forbucle como el uso inútil de catmediante algo como

$ cat input
A 19
B 81
C 76
D 31
$ awk '{system("echo "$2" "$1)}' input
19 A
81 B
76 C
31 D
$ 

¿Dónde echoestaría en cambio tu "mis órdenes"...?

Respuesta3

Una forma es configurarlo IFSen una nueva línea:

set -f; # suppress wildcard expansion since `cat...` is unquoted
IFS=$'\n'; for line in `cat file.txt`; do
   Am=`echo "$line" | awk '{print $1}'`
   Bm=`echo "$line" | awk '{print $2}'`
   #my command Am +Bm.......
done

Aunque también existen otros métodos:

while read Am Bm junk; do
   #my command Am +Bm....
done < file.txt

información relacionada