У меня есть файл script, который мне нужно изменить с помощью другого script, чтобы вставить текст на 8-й строке.
Строка для вставки: Project_Name=sowstest
, в файл с именем start
.
Я пытался использовать awk и sed, но моя команда искажается.
У меня есть файл script, который мне нужно изменить с помощью другого script, чтобы вставить текст на 8-й строке.
Строка для вставки: Project_Name=sowstest
, в файл с именем start
.
Я пытался использовать awk и sed, но моя команда искажается.
sed -i '8i8 This is Line 8' FILE
вставляет в строку 8
8 This is Line 8
в файл FILE
-i
делает изменения непосредственно файлу FILE, не выводит на stdout, как упоминалось в комментариях glenn jackman.
Ответ ed
ed file << END
8i
Project_Name=sowstest
.
w
q
END
.
на своей собственной линии заканчивает режим ввода; w
пишет; q
завершает работу. GNU ed имеет команду wq
для сохранения и выхода, но старые ed не работают.
Дополнительная литература: https://gnu.org/software/ed/manual/ed_manual.html -
ответ awk
awk -v n=8 -v s="Project_Name=sowstest" 'NR == n {print s} {print}' file > file.new
POSIX sed
(и, например, OS X sed
, sed
ниже) требуется i
, за которым следует обратная косая черта и новая строка. Также по крайней мере OS X sed
не включает в себя новую строку после вставленного текста:
$ 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
Чтобы заменить строку, вы можете использовать команды c
(change) или s
(substitute) с числовым адресом:
$ 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
Альтернативы, использующие 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
awk
интерпретирует обратную косую черту в переменных, переданных с помощью -v
, но не в переменных, переданных с помощью 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
Оба ENVIRON
и -v
определяются POSIX.
Для тех, кто находится на SunOS, который не является GNU, следующий код поможет:
sed '1i\^J
line to add' test.dat > tmp.dat
Решения Perl:
perl -lpe 'print "Project_Name=sowstest" if $. == 8' file
-l
стирает новые строки и добавляет их обратно, устраняя необходимость в "\n" -p
перемещается по входному файлу, распечатывая каждую строку-e
выполняет код в одинарных кавычках $.
- номер строки
perl -slpe 'print $s if $. == $n' -- -n=8 -s="Project_Name=sowstest" file
-s
позволяет рудиментарный парсер аргументов--
предотвращает анализ -n и -s стандартным парсером аргументов perl perl -lpe 'BEGIN{$n=shift; $s=shift}; print $s if $. == $n' 8 "Project_Name=sowstest" file
setenv n 8 ; setenv s "Project_Name=sowstest"
echo $n ; echo $s
perl -slpe 'print $ENV{s} if $. == $ENV{n}' file
ENV
- это хэш, содержащий все переменные среды
perl -MGetopt::Std -lpe 'BEGIN{getopt("ns",\%o)}; print $o{s} if $. == $o{n}' -- -n 8 -s "Project_Name=sowstest" file
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 - рекомендуемое решение для стандартной библиотеки.
Это может быть чрезмерным для однострочных perl-скриптов, но это можно сделать
sed
Флаг -i
работает по-разному в macOS sed
, чем в GNU sed
.
Здесь можно использовать его на macOS/OS X:
sed -i '' '8i\
8 This is Line 8' FILE
Подробнее см. man 1 sed
.
sed -e '8iProject_Name=sowstest' -i start
с помощью GNU sed
Пример прогона:
[[email protected] ~]# for ((i=1; i<=10; i++)); do echo "Line #$i"; done > a_file
[[email protected] ~]# cat a_file
Line #1
Line #2
Line #3
Line #4
Line #5
Line #6
Line #7
Line #8
Line #9
Line #10
[[email protected] ~]# sed -e '3ixxx inserted line xxx' -i a_file
[[email protected] ~]# 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$
[[email protected] ~]#