出于某种原因,我似乎找不到一个直截了当的答案,我现在有点时间紧张。如何使用sed命令在匹配特定字符串的第一行之后插入选择的文本行呢?我有……

CLIENTSCRIPT="foo"
CLIENTFILE="bar"

我想在CLIENTSCRIPT=行之后插入一行,导致…

CLIENTSCRIPT="foo"
CLIENTSCRIPT2="hello"
CLIENTFILE="bar"

当前回答

awk变体:

awk '1;/CLIENTSCRIPT=/{print "CLIENTSCRIPT2=\"hello\""}' file

其他回答

一个POSIX兼容的,使用s命令:

sed '/CLIENTSCRIPT="foo"/s/.*/&\
CLIENTSCRIPT2="hello"/' file

Sed命令,适用于MacOS(至少,OS 10)和Unix(即。不像Gilles(目前被接受)那样需要gnu sed):

sed -e '/CLIENTSCRIPT="foo"/a\'$'\n''CLIENTSCRIPT2="hello"' file

这可以在bash中工作,也许其他shell也知道$'\n'计算引用样式。所有东西都可以写在一条线上 旧的/POSIX sed命令。如果可能有多行匹配CLIENTSCRIPT="foo"(或类似的代码),并且您希望第一次只添加额外的行,您可以按以下方式重新工作:

sed -e '/^ *CLIENTSCRIPT="foo"/b ins' -e b -e ':ins' -e 'a\'$'\n''CLIENTSCRIPT2="hello"' -e ': done' -e 'n;b done' file

(这会在行插入代码之后创建一个循环,循环遍历文件的其余部分,再也不会回到第一个sed命令)。

您可能会注意到,我在匹配模式中添加了一个'^ *',以防该行出现在注释中,或者被缩进。它不是100%完美的,但涵盖了其他一些可能常见的情况。根据需要调整…

这两种解决方案还解决了一个问题(对于添加一行的通用解决方案),即如果新插入的行包含未转义的反斜杠或&号,sed将解释它们,结果可能不一样,就像\n是- eg一样。\0将是匹配的第一行。尤其方便的是,如果你添加的一行来自一个变量,否则你必须先转义所有的东西之前使用${var//},或另一个sed语句等。

这种解决方案在脚本中稍微不那么混乱(但是引号和\n不容易阅读),当您不想在一个有缩进行的函数中将a命令的替换文本放在一行的开头时。我已经利用了$'\n'被shell计算为换行,它不是常规的'\n'单引号值。

虽然它已经足够长了,但我认为perl/甚至awk可能会胜出,因为它更具可读性。

我有一个类似的任务,并不能得到上述perl解决方案的工作。

以下是我的解决方案:

我的名字叫perl - f

解释:

使用正则表达式搜索/etc/mysql/my.cnf文件中仅包含[mysqld]的一行,并将其替换为

(mysqld) Collation-server = utf8_unicode_ci

在包含[mysqld]的行之后有效地添加sort -server = utf8_unicode_ci行。

awk变体:

awk '1;/CLIENTSCRIPT=/{print "CLIENTSCRIPT2=\"hello\""}' file

注意标准sed语法(就像在POSIX中一样,所以所有符合sed的实现都支持(GNU, OS/X, BSD, Solaris…)):

sed '/CLIENTSCRIPT=/a\
CLIENTSCRIPT2="hello"' file

或者在一行上:

sed -e '/CLIENTSCRIPT=/a\' -e 'CLIENTSCRIPT2="hello"' file

(-expression(以及-files的内容)用换行符连接,组成sed解释的sed脚本)。

用于就地编辑的-i选项也是一个GNU扩展,一些其他实现(如FreeBSD的)支持-i”。

另外,为了可移植性,你可以使用perl:

perl -pi -e '$_ .= qq(CLIENTSCRIPT2="hello"\n) if /CLIENTSCRIPT=/' file

或者你可以用ed或ex:

printf '%s\n' /CLIENTSCRIPT=/a 'CLIENTSCRIPT2="hello"' . w q | ex -s file