Inserire una riga con un numero di riga specifico con sed o awk


146

Ho un file di script che devo modificare con un altro script per inserire un testo all'ottava riga.

Stringa da inserire:, Project_Name=sowstestin un file chiamato start.

Ho provato a usare awk e sed, ma il mio comando è confuso.

Risposte:


231
sed -i '8i8 This is Line 8' FILE

inserisce alla riga 8

8 This is Line 8

nel file FILE

-i esegue la modifica direttamente sul file FILE, nessun output su stdout, come indicato nei commenti di glenn jackman.


3
Sì, l'opzione -i è specifica per GNU-sed.
Utente sconosciuto il

1
No. -i significa "modifica il file specificato sul posto". L'inserimento rispetto all'appendice si ottiene con "8isomething" vs. "8asomething", indipendentemente dall'interruttore -i.
Utente sconosciuto il

11
utenti mac: con homebrew, brew install gnu-sedquindi gsed
usalo

4
Questo è super utile! Esiste un modo per inserire spazi all'inizio della riga? Ho notato che sed non sta prestando attenzione allo spazio bianco iniziale ...
elju,

9
@elju: Sì, mascherarlo con un backslash: sed '8i\ 8 This is Line 8' FILE.
utente sconosciuto


19

OS X / macOS / FreeBSD sed

Il -iflag funziona in modo diverso su macOS sedrispetto a GNU sed.

Ecco come usarlo su macOS / OS X:

sed -i '' '8i\
8 This is Line 8' FILE

Vedi man 1 sedper maggiori informazioni.


18

la risposta terribile

awk -v n=8 -v s="Project_Name=sowstest" 'NR == n {print s} {print}' file > file.new

@glenn jackman Devo entrare #define SERVER@"http://10.35.42.54/ms0.8"in una riga particolare. Come posso raggiungere questo obiettivo?
Nevin Raj Victor,

1
Penso che Nevin debba solo sfuggire alle virgolette nella sua stringa con barre rovesciate
Chris Koknat,

@waLLe, inizia con la pagina di informazioni di Awk che contiene una bella descrizione di come funziona Awk. Qui, ho 2 coppie "condition {action}", il 2 ° non ha condizioni, il che significa che l'azione viene eseguita per ogni record. Dopo aver finito di leggere e hai ancora domande, fammi sapere.
Glenn Jackman,

@glennjackman quasi lì .. proprio non capisco: file> file.new
w411 3

"file" rappresenta il nome del file su cui sta lavorando awk. >è il simbolo di reindirizzamento della shell in modo che l'output di awk sia memorizzato nel file denominato "file.new".
Glenn Jackman,

16

POSIX sed(e, ad esempio, OS X sed, di sedseguito) devono iessere seguiti da una barra rovesciata e da una nuova riga. Inoltre, almeno OS X sednon include una nuova riga dopo il testo inserito:

$ seq 3|gsed '2i1.5'
1
1.5
2
3
$ seq 3|sed '2i1.5'
sed: 1: "2i1.5": command i expects \ followed by text
$ seq 3|sed $'2i\\\n1.5'
1
1.52
3
$ seq 3|sed $'2i\\\n1.5\n'
1
1.5
2
3

Per sostituire una riga, è possibile utilizzare i comandi c(modifica) o s(sostituzione) con un indirizzo numerico:

$ seq 3|sed $'2c\\\n1.5\n'
1
1.5
3
$ seq 3|gsed '2c1.5'
1
1.5
3
$ seq 3|sed '2s/.*/1.5/'
1
1.5
3

Alternative che utilizzano awk:

$ seq 3|awk 'NR==2{print 1.5}1'
1
1.5
2
3
$ seq 3|awk '{print NR==2?1.5:$0}'
1
1.5
3

awkinterpreta le barre rovesciate nelle variabili passate -vma non nelle variabili passate usando ENVIRON:

$ seq 3|awk -v v='a\ba' '{print NR==2?v:$0}'
1
a
3
$ seq 3|v='a\ba' awk '{print NR==2?ENVIRON["v"]:$0}'
1
a\ba
3

Entrambi ENVIRONe -vsono definiti da POSIX.


7

Soluzioni Perl:

veloce e sporco:

perl -lpe 'print "Project_Name=sowstest" if $. == 8' file

  • -l rimuove le nuove righe e le aggiunge nuovamente, eliminando la necessità di "\ n"
  • -p scorre il file di input, stampando ogni riga
  • -e esegue il codice tra virgolette singole

$. è il numero di riga

equivalente alla soluzione awk di @ glenn, usando argomenti chiamati:

perl -slpe 'print $s if $. == $n' -- -n=8 -s="Project_Name=sowstest" file

  • -s abilita un parser di argomento rudimentale
  • -- impedisce a -n e -s di essere analizzato dal parser di argomenti standard perl

argomenti del comando posizionale:

perl -lpe 'BEGIN{$n=shift; $s=shift}; print $s if $. == $n' 8 "Project_Name=sowstest" file

variabili ambientali:

setenv n 8 ; setenv s "Project_Name=sowstest"
echo $n ; echo $s
perl -slpe 'print $ENV{s} if $. == $ENV{n}' file

ENV è l'hash che contiene tutte le variabili di ambiente

Getopt per analizzare gli argomenti nell'hash% o:

perl -MGetopt::Std -lpe 'BEGIN{getopt("ns",\%o)}; print $o{s} if $. == $o{n}' -- -n 8 -s "Project_Name=sowstest" file

Getopt :: Nomi di opzioni lunghi e più lunghi

perl -MGetopt::Long -lpe 'BEGIN{GetOptions(\%o,"line=i","string=s")}; print $o{string} if $. == $o{line}' -- --line 8 --string "Project_Name=sowstest" file

Getopt è la soluzione di libreria standard consigliata.
Questo può essere eccessivo per gli script perl a una riga, ma può essere fatto


2
Complimenti a te Chris per aver dedicato del tempo a spiegare tutto ciò e ad esporlo così bene, ma wow è per questo che non mi piace il perl.
rsaw

5

Per coloro che sono su SunOS che non è GNU, il seguente codice aiuterà:

sed '1i\^J
line to add' test.dat > tmp.dat
  • ^ J è inserito con ^ V + ^ J
  • Aggiungi la nuova riga dopo '1i.
  • \ DEVE essere l'ultimo carattere della riga.
  • La seconda parte del comando deve essere in una seconda riga.

5

sed -e '8iProject_Name=sowstest' -i start usando GNU sed

Esecuzione di esempio:

[root@node23 ~]# for ((i=1; i<=10; i++)); do echo "Line #$i"; done > a_file
[root@node23 ~]# cat a_file
Line #1
Line #2
Line #3
Line #4
Line #5
Line #6
Line #7
Line #8
Line #9
Line #10
[root@node23 ~]# sed -e '3ixxx inserted line xxx' -i a_file 
[root@node23 ~]# cat -An a_file 
     1  Line #1$
     2  Line #2$
     3  xxx inserted line xxx$
     4  Line #3$
     5  Line #4$
     6  Line #5$
     7  Line #6$
     8  Line #7$
     9  Line #8$
    10  Line #9$
    11  Line #10$
[root@node23 ~]# 
[root@node23 ~]# sed -e '5ixxx (inserted) "line" xxx' -i a_file
[root@node23 ~]# cat -n a_file 
     1  Line #1
     2  Line #2
     3  xxx inserted line xxx
     4  Line #3
     5  xxx (inserted) "line" xxx
     6  Line #4
     7  Line #5
     8  Line #6
     9  Line #7
    10  Line #8
    11  Line #9
    12  Line #10
[root@node23 ~]# 

Da dove viene il trailing $proveniente dalla riga 3 dopo l'inserimento?
17-17

È dalla -Abandiera al cat:)
jno

cosa succede se la stringa che si desidera inserire contiene virgolette, parentesi ecc.?
Anentropico

Vedi aggiornamento al testo sopra. Ciò significa che dovrai fuggire quei caratteri come appropriato.
gennaio

0

sed -i "" -e $ '4 a \\ n''Project_Name = sowstest' start

  • Questa linea funziona perfettamente in macOS
Utilizzando il nostro sito, riconosci di aver letto e compreso le nostre Informativa sui cookie e Informativa sulla privacy.
Licensed under cc by-sa 3.0 with attribution required.