ChatGPT解决这个技术问题 Extra ChatGPT

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

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

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

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


V
Vishrant
sed -i '8i This is Line 8' FILE

在第 8 行插入

This is Line 8

进入文件 FILE

-i 直接对文件 FILE 进行修改,不输出到标准输出,如 glenn jackman 的评论中所述。


是的,-i 开关是特定于 GNU-sed 的。
不。 -i 表示“就地修改指定文件”。插入与追加是通过“8isothing”与“8asomething”实现的,独立于-i-switch。
mac 用户:使用自制软件,brew install gnu-sed,然后将其与 gsed 一起使用
这个超级好用!无论如何我可以在行首插入空格吗?我注意到 sed 没有注意初始空格...
@elju:是的,用反斜杠掩盖它:sed '8i\ 8 This is Line 8' FILE
M
Mateusz Piotrowski

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


M
Mateusz Piotrowski

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 变量将展开。
g
glenn jackman

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" 输入到特定行。我怎样才能做到这一点?
我认为内文只需要用反斜杠转义字符串中的引号
@waLLe,从 awk info page 开始,它很好地描述了 awk 的工作原理。在这里,我有 2 个“条件 {action}”对,第二个没有条件,这意味着对每条记录执行操作。读完后还有问题,请告诉我。
@glennjackman 几乎就在那里.. 只是没有得到这个:file > file.new
“file”表示 awk 正在处理的文件的名称。 > 是 shell redirection 符号,因此 awk 的输出存储在名为“file.new”的文件中。
L
Lri

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==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 定义。


j
jno

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 :)
如果要插入的字符串包含引号、括号等怎么办?
请参阅上述文本的更新。这意味着,您必须酌情转义这些字符。
C
Chris Koknat

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 $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 脚本来说可能有点过分了,但可以做到


感谢你 Chris 花时间解释所有这些并把它布置得这么好,但是哇这就是我不喜欢 perl 的原因。
N
Nasri Najib

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

sed '1i\^J
line to add' test.dat > tmp.dat

^J 插入 ^V+^J

在 '1i 之后添加换行符。

\ 必须是该行的最后一个字符。

命令的第二部分必须在第二行。


b
buddemat

在 linux 中添加 2 行可以正常工作。

sed '2s/$/ myalias/' file

u
umläute
sed -i "" -e $'4 a\\\n''Project_Name=sowstest' start

此行在 macOS 中运行良好


I
IAng

谢谢你umläute

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

以下在 macOS 上很有用,可以在 4 之后添加新行

为了循环,我创建了一个文件夹数组,在 mac zsh 中迭代它们

for foldercc in $foldernames; 

sed -i "" -e $'4 a\\\n''Project_Name=sowstest' $foldercc/filenames;