如何使用 sed 在文件中查找和替换的字符串

通常在处理文本文件时,您需要在一个或多个文件中查找和替换文本字符串。

sed 是一个 stream  editor 。它可以对文件和输入流(如管道)执行基本文本操作。有了 sed 你可以搜索,查找和替换,插入和删除文字和字符串。它支持正则表达式,允许您匹配复杂的模式。

在本文中,我们将讨论如何使用 sed 查找和替换字符串。我们还将向您展示如何执行递归搜索和替换。

查找和替换字符串

sed 有几个版本,它们之间存在一些功能差异。 MacOS 使用 BSD 版本,大多数 Linux 发行版都默认预装了 GNU sed 。我们将使用 GNU 版本。

sed 搜索和替换文本的一般采用以下形式:

sed -i 's/SEARCH_REGEX/REPLACEMENT/g' INPUTFILE
  • -i- 默认情况下 sed 将其输出写入标准输出。此选项告诉 sed 直接编辑文件。如果提供了扩展名(例如 -i.bak) ,则将创建原始文件的备份。
  • s  - 替换命令,可能是 sed 中最常用的命令。
  • /// - 分隔符。它可以是任何字符,但通常使用斜杠 (/) 字符。
  • SEARCH_REGEX - 正常字符串或要搜索的正则表达式。
  • REPLACEMENT - 替换字符串。
  • g - 全局替换标志。默认情况下, sed 逐行读取文件并仅更改每行第一次出现的 SEARCH_REGEX。当使用 g 替换标志时,将替换所有的匹配项。
  • INPUTFILE - 要处理的文件的名称。

在参数周围加上引号是一个很好的做法,因此 shell 元字符不会扩展。

让我们看看如何使用 sed 命令搜索和替换文件中的文本及其一些最常用的选项和标志的示例。

出于演示目的,我们将使用以下 file.txt 文件:

123 Foo foo foo 
foo /bin/bash Ubuntu foobar 456

如果省略该 g 标志,则只替换每行中搜索字符串的第一个实例:

sed -i 's/foo/linux/' file.txt
123 Foo linux foo 
linux /bin/bash Ubuntu foobar 456

使用全局替换标志 sed 替换所有的匹配项:

sed -i 's/foo/linux/g' file.txt
123 Foo linux linux
linux /bin/bash Ubuntu linuxbar 456

正如你可能已经注意到,在前面的例子的 foobar 的子串 foo 也被替换。如果这不是想要的行为,请在搜索字符串的两端使用单词边界表达式(\b) 。这可确保部分单词不匹配。

sed -i 's/\bfoo\b/linux/g' file.txt
123 Foo linux linux
linux /bin/bash Ubuntu foobar 456

要使模式匹配不区分大小写,请使用该 I 标志。在下面的示例中,我们使用 gI 标志:

sed -i 's/foo/linux/gI' file.txt
123 linux linux linux 
linux /bin/bash Ubuntu linuxbar 456

如果要查找并替换包含分隔符 / 的字符串,则需要使用反斜杠 (\) 来转义斜杠。例如,以取代 /bin/bash/usr/bin/zsh 你会使用

sed -i 's/\/bin\/bash/\/usr\/bin\/zsh/g' file.txt

使用另一个分隔符 | 也许更易读一些。大多数人使用竖线 (|) 或冒号 (:) ,您也可以使用任何其他字符:

sed -i 's|/bin/bash|/usr/bin/zsh|g' file.txt
123 Foo foo foo 
foo /usr/bin/zsh Ubuntu foobar 456

您还可以使用正则表达式。例如,要搜索所有3位数字并将其替换为 number,请看下面的命令:

sed -i 's/\b[0-9]\{3\}\b/number/g' file.txt
number Foo foo foo 
foo /bin/bash demo foobar number

sed 的另一个有用功能是你可以使用 & 与匹配项对应替换。

例如,如果要 {} 在每个 3 位数字周围添加花括号,请使用下面的命令:

sed -i 's/\b[0-9]\{3\}\b/{&}/g' file.txt
{123} Foo foo foo 
foo /bin/bash demo foobar {456}

最后一点说一下,在使用 sed 编辑文件时进行备份是很有必要的。要做到这一点,只需提供 -i 选项的扩展。例如,要编辑 file.txt 并保存原始文件的备份 file.txt.bak ,如下所示:

sed -i.bak 's/foo/linux/g' file.txt

如果要确保创建备份,请使用以下 ls 命令列出文件:

ls
file.txt file.txt.bak

递归查找和替换

有时,您希望以递归方式在目录中搜索包含字符串的文件,并替换所有文件中的字符串。这可以通过配合其他命令来完成。使用 find 或者 grep 以递归方式查找目录中的文件并将文件名汇总到 sed

以下命令将递归搜索当前工作目录中的文件并将文件名传递给 sed

find . -type f -exec sed -i 's/foo/bar/g' {} +

为避免名称中包含空格的文件出现问题,请使用 -print0 告诉 find 打印文件名的选项,后跟空字符并通过管道使用 xargs -0 输出到 sed

find . -type f -print0 | xargs -0 sed -i 's/foo/bar/g'

要排除目录,请使用该 -not -path 选项。例如,如果要替换本地 git 仓库中的除以点 (.) 开头的所有文件,请使用:

find . -type f -not -path '*/\.*' -print0 | xargs -0 sed -i 's/foo/bar/g'

如果您只想在具有特定扩展名的文件上搜索和替换文本,您可以使用:

find . -type f -name "*.md" -print0 | xargs -0 sed -i 's/foo/bar/g'

另一种选择是使用该 grep 命令递归查找包含搜索模式的所有文件,然后将文件名传递给 sed

grep -rlZ 'foo' . | xargs -0 sed -i.bak 's/foo/bar/g'

结论

虽然看起来很复杂,但使用 sed 搜索和替换文件中的文本的几个常用选项又非常简单。

要了解有关 sed 命令,选项和标志的更多信息,请访问 GNU sed 手册Grymoire sed 教程。如果您有任何问题或反馈,请随时发表评论。