使用sed或awk在特定行号插入一行

220

我有一个脚本文件,需要用另一个脚本来修改,在第8行插入文本。

要插入的字符串:Project_Name=sowstest,插入到名为start的文件中。

我尝试使用awk和sed命令,但是我的命令变得混乱了。

12个回答

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

在第8行插入

This is Line 8

将内容写入文件FILE中

-i选项会直接修改文件FILE,不会输出到标准输出流(stdout),正如glenn jackman在评论中所提到的。


3
是的,-i选项是专为GNU-sed设计的。 - user unknown
2
-i选项表示“就地修改指定文件”。插入和追加可以通过使用'8isomething'和'8asomething'来实现,与-i开关无关。 - user unknown
22
Mac用户:使用Homebrew安装gnu-sed,然后使用gsed命令。 - cwd
4
非常有用!我是否有办法在行首插入空格?我注意到 sed 不会注意初始的空格... - elju
11
是的,可以用反斜杠作为掩码: sed '8i\ 8 This is Line 8' FILE - user unknown
显示剩余10条评论

38

OS X / macOS / FreeBSD sed

-i 标志在 macOS 的 sed中与 GNU 的 sed有所不同。

以下是在 macOS / OS X 上使用它的方法:

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

请参阅man 1 sed以获取更多信息。


能否使用变量而不是字符串字面值内联来完成这个操作? - Andy Ray
1
@AndyRay 使用双引号代替单引号。这样shell变量就会被扩展。 - Mateusz Piotrowski
在 macOS 上为文件添加新行,请使用以下命令:sed -i '' '8i\ 8 This is Line 8\' FILE - xgqfrms

36
一个 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


22

awk答案

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

@glenn jackman 我需要将 #define SERVER@"http://10.35.42.54/ms0.8" 输入到特定行。我该如何做到这一点? - Nevin Raj Victor
1
我认为Nevin只需要用反斜杠转义字符串中的引号。 - Chris Koknat
"file" 代表 awk 正在处理的文件名。> 是 shell 重定向 符号,以便将 awk 的输出存储在名为 "file.new" 的文件中。 - glenn jackman
@glennjackman。NR == n 映射行号,{print s} 将在第8行打印变量s。然后{print}将打印文件中的所有记录,然后您可以使用file.new作为重定向。我的理解正确吗? - jian
多少而已。我会说"NR == n 测试 行号"。然后,当当前记录数为8时,打印变量。{print}无条件地打印每一行。欲了解更多信息,请点击问题下方的 [tag:awk] 标签,然后点击 "Learn more…" 链接。 - glenn jackman
显示剩余2条评论

18

POSIX标准下的sed命令(例如OS X中的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(更改)或s(替换)命令:

$ 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定义的。


11

使用GNU sed:sed -e '8iProject_Name=sowstest' -i start

示例运行:

[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 ~]# 

在插入后的第3行,尾随的 $ 是从哪里来的? - jww
-A 标志到 cat :) - jno
如果要插入的字符串包含引号、括号等符号怎么办? - Anentropic
这意味着您将不得不适当地转义那些字符。 - jno

7

Perl解决方案:

快速且简单:

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

  • -l去除换行符并重新添加,消除了"\n"的需要
  • -p循环遍历输入文件,打印每一行
  • -e执行单引号中的代码

$.是行号

等同于@glenn的awk解决方案,使用命名参数:

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

  • -s启用一个基本的参数解析器
  • --防止标准perl参数解析器解析-n和-s

位置命令参数:

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 是包含所有环境变量的哈希表

使用 Getopt 将参数解析为哈希表 %o:

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

使用 Getopt::Long 和更长的选项名称

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 脚本来说,这可能有些过度,但是它确实可以做到。


3
Chris,感谢你花时间解释这一切并将其清晰地阐述出来,但是哇,这就是为什么我不喜欢Perl。 - rsaw

5

对于那些使用非GNU的SunOS系统的人,以下代码将会有所帮助:

sed '1i\^J
line to add' test.dat > tmp.dat
  • 插入^J需要使用^V+^J
  • 在“1i.”后添加换行符
  • \必须是行末的最后一个字符
  • 命令的第二部分必须在第二行中

1
sed -i "" -e $'4 a\\\n''Project_Name=sowstest' start
  • 这行在 macOS 上运行良好

1

macOS sed 解决方案

例如:在第一行插入

  1. ns
# recommended 
# This command only needs to write one line
$ sed -i '' '1s/^/The new First Line\n/' ./your-source-file-name

2. ni 是指在一个半导体材料中掺入杂质后形成的杂质离子数目。
# not recommended 
# This way the command needs to be written on multiple lines
$ sed -i '' '1i\
The new First Line\
' ./your-source-file-name

测试演示

$ sed -i '' '1s/^/Perl  camel\n/' ./multi-line-text.txt

enter image description here


这个推荐的解决方案在Makefile中运行得非常好! - RexYuan

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接