No Linux eu tenho o arquivo orig-file.txt
. Este arquivo agora inclui 4 campos, mas podem ser menos ou mais (este arquivo é gerado por outro aplicativo).
Qual é a melhor opção para traduzir orig-file.txt
para um arquivo semelhante a um output-file.txt
arquivo (pode ser com um script Shell ou awk, etc.)
arquivo original.txt
CREATE_TIMESTAMP TELEPHONE_NUMBER ID TYPE
------------------- -------------------- ---------- -----------------
24-09-2009 16:17:45 33633333333 20 other_mmm_phone
24-09-2009 17:45:07 33644444444 20 other_mmm_phone
07-10-2009 10:45:49 12312312312 20 legacyphone
07-10-2009 11:46:38 59320000043 20 other_mmm_phone
arquivo de saída.txt
CREATE_TIMESTAMP -> 24-09-2009 16:17:45
TELEPHONE_NUMBER -> 33633333333
ID -> 20
TYPE -> other_mmm_phone
---
CREATE_TIMESTAMP -> 24-09-2009 16:17:45
TELEPHONE_NUMBER -> 33633333333
ID -> 20
TYPE -> other_mmm_phone
---
exemplo de awk lang -( mas não funciona -:(
# awk 'NR>2 {
> printf "\
> %-16s -> %s\n\
> %-16s -> %s\n\
> %-16s -> %s\n\
> %-16s -> %s\
> \n\n\n---\n\n\n",\
> "CREATE_TIMESTAMP", $1" "$2,\
> "TELEPHONE_NUMBER", $3,\
> "ID", $4,\
> "TYPE", $5}\
> ' orig-file.txt
awk: newline in string near line 2
awk: syntax error near line 3
awk: illegal statement near line 3
awk: newline in string near line 7
Responder1
Aqui estão alguns ksh simples:
{
read t1 t2 t3 t4
maxlen=$(printf "%s\n" ${#t1} ${#t2} ${#t3} ${#t4} | sort -n | tail -1)
fmt=$(printf "%%-%ds -> %%s" $maxlen)
read line
while read date time tel id type; do
printf "$fmt\n" $t1 "$date $time" $t2 $tel $t3 $id $t4 $type
print "\n\n\n---\n\n"
done
} < orig-file.txt
Atualizarpara um número flexível de campos:
Eu substituo o espaço no campo data e hora para facilitar a análise
sed '3,$s/ /@@/' orig-file.txt |
{
read line
set -A headings $line
max=0
for head in "${headings[@]}"; do (( max < ${#head} )) && max=${#head}; done
fmt=$(printf "%%-%ds -> %%s" $max)
read line
while read line; do
set -A fields $line
i=0
while (( i < ${#headings[@]} )); do
printf "$fmt\n" ${headings[$i]} ${fields[$i]} | sed 's/@@/ /'
(( i=i+1 ))
done
print "\n\n\n---\n\n"
done
}
Responder2
Isso faz o trabalho neste caso. Modificação trivial necessária se mais campos forem adicionados.
awk 'NR>2{
printf "\
%-16s -> %s\n\
%-16s -> %s\n\
%-16s -> %s\n\
%-16s -> %s\
\n\n\n---\n\n\n",\
"CREATE_TIMESTAMP", $1" "$2,\
"TELEPHONE_NUMBER", $3,\
"ID", $4,\
"TYPE", $5}\
' orig-file.txt > output-file.txt
"CREATE_TIMESTAMP" precisa de ambos $1
e $2
já que a data em si é separada por espaços em branco.
Ele pode ser modificado para ler os nomes dos campos do cabeçalho, mas há o problema de a data ser separada por espaços em branco. Se outros campos também puderem conter espaços em branco, serão sempre necessárias modificações manuais para compensar, como no $1" "$2
caso.