使用 sed 或 awk 在特定行号处插入一行
Posted
技术标签:
【中文标题】使用 sed 或 awk 在特定行号处插入一行【英文标题】:Insert a line at specific line number with sed or awk 【发布时间】:2022-01-18 11:29:56 【问题描述】:我有一个脚本文件,我需要用另一个脚本修改它以在第 8 行插入文本。
要插入的字符串:Project_Name=sowstest
,插入到名为 start
的文件中。
我尝试使用 awk 和 sed,但我的命令出现乱码。
【问题讨论】:
【参考方案1】:sed -i '8i This is Line 8' FILE
在第 8 行插入
This is Line 8
进入文件 FILE
-i
直接对文件 FILE 进行修改,不输出到标准输出,正如 glenn jackman 在 cmets 中提到的那样。
【讨论】:
是的,-i 开关是特定于 GNU-sed 的。 没有。 -i 表示“就地修改指定文件”。插入与追加是通过“8isothing”与“8asomething”实现的,独立于-i-switch。 mac 用户:使用自制软件,brew install gnu-sed
,然后使用 gsed
这个超级有用!无论如何我可以在行首插入空格吗?我注意到 sed 没有注意初始空格...
@elju:是的,用反斜杠掩盖它:sed '8i\ 8 This is Line 8' FILE
。【参考方案2】:
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 只需要用反斜杠转义字符串中的引号
@waLLe,从awk info page 开始,它很好地描述了 awk 的工作原理。在这里,我有 2 个“条件 action”对,第二个没有条件,这意味着对每条记录执行操作。读完后仍有疑问,请告诉我。
@glennjackman 几乎就在那里.. 只是没有得到这个:file > file.new
"file" 表示 awk 正在处理的文件的名称。 >
是 shell redirection 符号,因此 awk 的输出存储在名为“file.new”的文件中。【参考方案3】:
ed
的回答
ed file << END
8i
Project_Name=sowstest
.
w
q
END
.
单独一行结束输入模式; w
写; q
退出。 GNU ed 有一个wq
命令来保存和退出,但旧版没有。
延伸阅读:https://gnu.org/software/ed/manual/ed_manual.html
【讨论】:
【参考方案4】:对于那些使用非 GNU 的 SunOS 的人,以下代码会有所帮助:
sed '1i\^J
line to add' test.dat > tmp.dat
^J 插入 ^V+^J
在 '1i 之后添加换行符。
\ 必须是该行的最后一个字符。
命令的第二部分必须在第二行中。
【讨论】:
【参考方案5】: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
(更改)或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==2print 1.51'
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 定义。
【讨论】:
【参考方案6】: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
启用基本参数解析器
--
防止 -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 $ENVs if $. == $ENVn' file
ENV
是包含所有环境变量的哈希
Getopt 将参数解析为哈希 %o:
perl -MGetopt::Std -lpe 'BEGINgetopt("ns",\%o); print $os if $. == $on' -- -n 8 -s "Project_Name=sowstest" file
Getopt::更长的选项名称
perl -MGetopt::Long -lpe 'BEGINGetOptions(\%o,"line=i","string=s"); print $ostring if $. == $oline' -- --line 8 --string "Project_Name=sowstest" file
Getopt 是推荐的标准库解决方案。 这对于单行 perl 脚本来说可能有点矫枉过正,但可以做到
【讨论】:
感谢 Chris,感谢您花时间解释所有这些并把它布置得这么好,但这就是我不喜欢 perl 的原因。【参考方案7】: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
了解更多信息。
【讨论】:
是否可以使用变量而不是字符串文字内联来做到这一点? @AndyRay 请改用双引号。然后shell变量将展开。【参考方案8】:sed -e '8iProject_Name=sowstest' -i start
使用 GNU sed
示例运行:
[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 行的尾随$
在哪里?
从-A
标志到cat
:)
如果要插入的字符串包含引号、括号等怎么办?
查看以上文字的更新。这意味着,您必须适当地转义这些字符。【参考方案9】:
sed -i "" -e $'4 a\\\n''Project_Name=sowstest' start
此行在 macOS 中运行良好
【讨论】:
【参考方案10】:添加 2 行在 linux 中运行良好。
sed '2s/$/ myalias/' file
【讨论】:
以上是关于使用 sed 或 awk 在特定行号处插入一行的主要内容,如果未能解决你的问题,请参考以下文章
使用 sed / awk / bash 将缺失的行号填充到文件中