Wstaw linię pod określonym numerem za pomocą sed lub awk

146

Mam plik skryptu, który muszę zmodyfikować za pomocą innego skryptu, aby wstawić tekst w 8. wierszu.

Ciąg do wstawienia:, Project_Name=sowstestdo pliku o nazwie start.

Próbowałem użyć awk i sed, ale moje polecenie jest zniekształcane.

ashok
źródło

Odpowiedzi:

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

wstawia w linii 8

8 This is Line 8

do pliku FILE

-i dokonuje modyfikacji bezpośrednio w pliku PLIK, bez wyjścia na standardowe wyjście, jak wspomniał w komentarzach glenn jackman.

nieznany użytkownik
źródło
3
Tak, przełącznik -i jest specyficzny dla GNU-sed.
użytkownik nieznany
1
Nie. -I oznacza „zmodyfikuj określony plik w miejscu”. Wstaw vs. dołączanie uzyskuje się za pomocą opcji „8isomething” vs. „8asomething”, niezależnie od przełącznika -i.
użytkownik nieznany
11
użytkownicy mac: z homebrew, brew install gnu-seda następnie używaj tego zgsed
cwd
4
To jest bardzo przydatne! Czy mimo wszystko mogę wstawić spacje na początku wiersza? Zauważyłem, że sed nie zwraca uwagi na początkowe białe spacje ...
elju
9
@elju: Tak, maskować go odwrotnym ukośnikiem: sed '8i\ 8 This is Line 8' FILE.
użytkownik nieznany
32

edodpowiedź

ed file << END
8i
Project_Name=sowstest
.
w
q
END

.na własnej linii kończy tryb wprowadzania; wpisze; qskwitowany. GNU ed ma wqpolecenie zapisywania i zamykania, ale stare edy nie.

Więcej informacji: https://gnu.org/software/ed/manual/ed_manual.html

glenn jackman
źródło
19

OS X / macOS / FreeBSD sed

-iFlag działa inaczej na MacOS sedniż w GNUsed .

Oto sposób korzystania z niego w systemie MacOS / OS X:

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

Zobacz, man 1 sedaby uzyskać więcej informacji.

Mateusz Piotrowski
źródło
18

odpowiedź awk

awk -v n=8 -v s="Project_Name=sowstest" 'NR == n {print s} {print}' file > file.new
glenn jackman
źródło
@glenn jackman Muszę wejść #define SERVER@"http://10.35.42.54/ms0.8"do określonej linii. Jak mogę to osiągnąć?
Nevin Raj Victor
1
Myślę, że Nevin musi po prostu uciec przed cudzysłowami w swoim ciągu z ukośnikami odwrotnymi
Chris Koknat
@waLLe, zacznij od strony informacyjnej awk, która zawiera ładny opis tego, jak działa awk. Tutaj mam 2 pary „warunek {akcja}”, druga nie ma warunku, co oznacza, że ​​akcja jest wykonywana dla każdego rekordu. Gdy skończysz czytać i nadal będziesz mieć pytania, daj mi znać.
glenn jackman
@glennjackman prawie gotowe ... po prostu nie dostałem tego: file> file.new
w411 3
„plik” reprezentuje nazwę pliku, nad którym pracuje awk. >jest symbolem przekierowania powłoki, dzięki czemu wyjście awk jest przechowywane w pliku o nazwie „plik.new”.
glenn jackman
16

POSIX sed(i na przykład OS X sed, sedponiżej) wymagają iodwrotnego ukośnika i nowej linii. Ponadto przynajmniej OS X sednie zawiera nowej linii po wstawionym tekście:

$ 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

Aby zamienić linię, możesz użyć poleceń c(zmień) lub s( zastąp ) z adresem numerycznym:

$ 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

Alternatywy wykorzystujące 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

awkinterpretuje ukośniki odwrotne w zmiennych przekazywanych, -vale nie w zmiennych przekazywanych przy użyciu 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

Obie ENVIRONi -vsą zdefiniowane przez POSIX.

Lri
źródło
7

Rozwiązania Perl:

szybko i brudno:

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

  • -l usuwa nowe linie i dodaje je z powrotem, eliminując potrzebę „\ n”
  • -p zapętla plik wejściowy, wypisując każdy wiersz
  • -e wykonuje kod w apostrofach

$. to numer linii

odpowiednik rozwiązania awk @ glenn, używając nazwanych argumentów:

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

  • -s włącza podstawowy parser argumentów
  • -- zapobiega analizowaniu opcji -n i -s przez standardowy parser argumentów perla

pozycyjne argumenty poleceń:

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

Zmienne środowiska:

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

ENV to skrót zawierający wszystkie zmienne środowiskowe

Getopt, aby przeanalizować argumenty w hash% o:

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

Getopt :: Długie i dłuższe nazwy opcji

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 to zalecane rozwiązanie w zakresie bibliotek standardowych.
Może to być przesada w przypadku jednowierszowych skryptów perl, ale można to zrobić

Chris Koknat
źródło
2
Chwała ci Chris za poświęcenie czasu na wyjaśnienie tego wszystkiego i ułożenie tego tak ładnie, ale wow, właśnie dlatego nie lubię perla.
rsaw
5

Dla tych, którzy korzystają z SunOS, który nie jest GNU, pomoże następujący kod:

sed '1i\^J
line to add' test.dat > tmp.dat
  • ^ J jest wstawiane z ^ V + ^ J
  • Dodaj nową linię po „1i.
  • \ MUSI być ostatnim znakiem w linii.
  • Druga część polecenia musi znajdować się w drugiej linii.
Nasri Najib
źródło
5

sed -e '8iProject_Name=sowstest' -i start używając seda GNU

Przykładowy przebieg:

[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 ~]# 
jno
źródło
Gdzie znajduje się zakończenie $pochodzące z linii 3 po wstawieniu?
jww
To od -Aflagi do cat:)
jno
co, jeśli ciąg, który chcesz wstawić, zawiera cudzysłowy, nawiasy itp.?
Anentropic
Zobacz aktualizację tekstu powyżej. Oznacza to, że będziesz musiał odpowiednio uciec od tych znaków.
jno
0

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

  • Ta linia działa dobrze w systemie MacOS
Elza James
źródło