it-swarm-pt.tech

Remover caracteres de fim de linha do stdout? Várias linhas em uma única linha

Eu tenho um script que produz o seguinte texto. Esta é a saída de um modem Netopia 2210-02 ADSL2 .

ADSL Line State:        Up
ADSL Startup Attempts:  1
ADSL Modulation:        DMT
ADSL Data Path:         Fast
Datapump Version:       DSP 7.2.3.0, HAL 7.2.1.0
SNR Margin:                   8.20        9.00 dB
Line Attenuation:            57.50       31.00 dB
Output Power:                17.09       12.34 dBm
Errored Seconds:                 0           0
Loss of Signal:                  0         476
Loss of Frame:                   0           0
CRC Errors:                  57921         416
Data Rate:                    2880        1024

Como posso remover o caractere de fim de linha de cada linha? Eu gostaria que a saída fosse parecida com isto (sim, é feio):

ADSL Line State:        Up ADSL Startup Attempts:  1 ADSL Modulation:        DMT ADSL Data Path:         Fast Datapump Version:       DSP 7.2.3.0, HAL 7.2.1.0 SNR Margin:                   8.20        9.00 dB Line Attenuation:            57.50       31.00 dB Output Power:                17.09       12.34 dBm Errored Seconds:                 0           0 Loss of Signal:                  0         476 Loss of Frame:                   0           0 CRC Errors:                  57921         416 Data Rate:                    2880        1024 

Tentei algumas soluções como esta, mas não funcionam:

# (This simply outputs the contents of the script, unmodified)
[email protected]:~/Work/Cacti $ ./script | sed -e 's/$//'

Também tentei usar tr. Eu esperava que o seguinte comando substituísse cada caractere de nova linha pelo caractere de espaço. Isso pegaria as várias linhas e as combinaria em uma única linha longa. Em vez disso, isso exibe apenas a última linha de saída. Parece substituir cada linha subsequente com a próxima linha de saída.

[email protected]:~/Work/Cacti $ ./script | tr '\n' ' '
Data Rate:                    2880        [email protected]:~/Work/Cacti $
[email protected]:~/Work/Cacti $

Atualizar :

Após um exame mais aprofundado, parece que cada linha é precedida por um caractere de retorno. Isso aparece como ^M ao usar less. Portanto, adicionei duas instruções tr. Um para excluir caracteres de nova linha, outro para excluir o caractere de retorno.

./script | | tr -d '\n' | tr -d '\r'
15
Stefan Lasiewski

sed não funcionará (facilmente) porque opera nas linhas uma de cada vez; você poderia fazer isso, mas envolveria copiar toda a entrada no buffer de retenção

tr realmente deve funcionar da maneira que você colou; tem certeza de que as novas linhas são \ns? Você pode simplificar um pouco excluindo as novas linhas em vez de convertê-las em espaços:

tr -d '\n'
18
Michael Mrozek

além de tr, alguns outros métodos que você pode usar

awk

awk '1' ORS= file

Concha

while read -r line; do printf "%s " "$line"; done <file

sed

sed -e ':a' -e 'N;s/\n/ /;ba' file
5
user1606

Usar tr -s '\n' para substituir sequências de caracteres de nova linha repetidos por uma única ocorrência de nova linha.

Exemplo:

>>> echo -e "Squeeze\n\n\nthose\n\n\n\nnewlines\n"
Squeeze


those



newlines

2z [franck:~] $ 
>>> echo -e "Squeeze\n\n\nthose\n\n\n\nnewlines\n" | tr -s '\n'
Squeeze
those
newlines

(Vejo man tr)

0
Franck

Para remover caracteres de nova linha, é tr -d '\n' < file. No entanto, para juntar todas as linhas, isso é paste -sd '\0' file.

tr -d '\n' produz uma saída não textual, pois não termina uma linha com um caractere de nova linha.

Para remover todos os caracteres CR, você pode fazer:

tr -d '\r' < file | paste -sd '\0' -

Para remover apenas os caracteres CR que estão no final das linhas:

CR=$(printf '\r')
sed "s/$CR\$//" < file | paste -sd '\0' -

Ou use awk para tudo:

awk '{gsub(/\r$/, ""); printf "%s", $0} 
     END {if (NR) print ""}' < file

Ou use dos2unix (o que removeria os CRs finais e também corrigiria alguns outros problemas com arquivos de texto de sistemas operacionais da Microsoft):

 dos2unix < file | paste -sd '\0' -
0
Stéphane Chazelas