Insérer une ligne à un numéro de ligne spécifique avec sed ou awk

146

J'ai un fichier de script que je dois modifier avec un autre script pour insérer un texte à la 8ème ligne.

Chaîne à insérer:, Project_Name=sowstestdans un fichier appelé start.

J'ai essayé d'utiliser awk et sed, mais ma commande est déformée.

ashok
la source

Réponses:

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

inserts à la ligne 8

8 This is Line 8

dans le fichier FILE

-i effectue la modification directement dans le fichier FILE, pas de sortie vers stdout, comme mentionné dans les commentaires de glenn jackman.

Utilisateur inconnu
la source
3
Oui, le commutateur -i est spécifique à GNU-sed.
utilisateur inconnu
1
Non. -I signifie "modifier le fichier spécifié sur place". L'insertion par rapport à l'ajout est réalisée avec '8isomething' contre '8asomething', indépendamment du commutateur -i.
utilisateur inconnu
11
utilisateurs mac: avec homebrew, brew install gnu-sedpuis utilisez-le avecgsed
cwd
4
C'est super utile! Y a-t-il de toute façon pour moi d'insérer des espaces au début de la ligne? J'ai remarqué que sed ne
faisait
9
@elju: Oui, le masque avec une barre oblique inverse: sed '8i\ 8 This is Line 8' FILE.
utilisateur inconnu le
32

Une edréponse

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

.sur sa propre ligne termine le mode d'entrée; wécrit; qquitte. GNU ed a une wqcommande pour sauvegarder et quitter, mais pas les anciens ed.

Lectures complémentaires: https://gnu.org/software/ed/manual/ed_manual.html

Glenn Jackman
la source
19

OS X / macOS / FreeBSD sed

L' -iindicateur fonctionne différemment sur macOS sedque dans GNU sed.

Voici comment l'utiliser sur macOS / OS X:

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

Voir man 1 sedpour plus d'informations.

Mateusz Piotrowski
la source
18

la réponse awk

awk -v n=8 -v s="Project_Name=sowstest" 'NR == n {print s} {print}' file > file.new
Glenn Jackman
la source
@glenn jackman J'ai besoin d'entrer #define SERVER@"http://10.35.42.54/ms0.8"sur une ligne particulière. Comment puis-je atteindre cet objectif?
Nevin Raj Victor
1
Je pense que Nevin a juste besoin d'échapper aux guillemets dans sa chaîne avec des barres obliques inverses
Chris Koknat
@waLLe, commencez par la page d'informations awk qui a une belle description du fonctionnement de awk. Ici, j'ai 2 paires "condition {action}", la 2ème n'a aucune condition ce qui signifie que l'action est effectuée pour chaque enregistrement. Après avoir fini de lire et que vous avez encore des questions, faites-le moi savoir.
glenn jackman
@glennjackman presque là ... je n'ai tout simplement pas compris ceci: file> file.new
w411 3
"fichier" représente le nom du fichier sur lequel awk travaille. >est le symbole de redirection du shell pour que la sortie de awk soit stockée dans le fichier nommé "file.new".
glenn jackman
16

POSIX sed(et par exemple OS X sed, sedci - dessous) idoivent être suivis d'une barre oblique inverse et d'une nouvelle ligne. De plus, au moins OS X sedn'inclut pas de nouvelle ligne après le texte inséré:

$ 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

Pour remplacer une ligne, vous pouvez utiliser les commandes c(modifier) ​​ou s(remplacer) par une adresse numérique:

$ 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

Alternatives utilisant 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

awkinterprète les barres obliques inverses dans les variables passées avec -vmais pas dans les variables passées en utilisant 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

Les deux ENVIRONet -vsont définis par POSIX.

Lri
la source
7

Solutions Perl:

rapide et sale:

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

  • -l supprime les nouvelles lignes et les rajoute, éliminant ainsi le besoin de "\ n"
  • -p boucle sur le fichier d'entrée, imprimant chaque ligne
  • -e exécute le code entre guillemets simples

$. est le numéro de ligne

équivalent à la solution awk de @ glenn, utilisant des arguments nommés:

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

  • -s active un analyseur d'arguments rudimentaire
  • -- empêche -n et -s d'être analysés par l'analyseur d'arguments perl standard

arguments de commande de position:

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

Variables d'environnement:

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

ENV est le hachage qui contient toutes les variables d'environnement

Getopt pour analyser les arguments dans le hachage% o:

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

Getopt :: Noms d'options longs et longs

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 est la solution de bibliothèque standard recommandée.
Cela peut être excessif pour les scripts perl sur une ligne, mais cela peut être fait

Chris Koknat
la source
2
Félicitations à toi Chris pour avoir pris le temps d'expliquer tout cela et de le présenter si bien, mais wow c'est pourquoi je n'aime pas perl.
rsaw
5

Pour ceux qui utilisent SunOS qui n'est pas GNU, le code suivant aidera:

sed '1i\^J
line to add' test.dat > tmp.dat
  • ^ J est inséré avec ^ V + ^ J
  • Ajoutez la nouvelle ligne après '1i.
  • \ DOIT être le dernier caractère de la ligne.
  • La deuxième partie de la commande doit être dans une deuxième ligne.
Nasri Najib
la source
5

sed -e '8iProject_Name=sowstest' -i start en utilisant GNU sed

Exemple d'exécution:

[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
la source
D'où vient la fin $de la ligne 3 après l'insertion?
jww
C'est du -Adrapeau à cat:)
jno
que faire si la chaîne que vous souhaitez insérer contient des guillemets, des parenthèses, etc.?
Anentropic
Voir la mise à jour du texte ci-dessus. Cela signifie que vous devrez échapper à ces caractères le cas échéant.
jno
0

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

  • Cette ligne fonctionne bien dans macOS
Elza James
la source