如何使用 sed 查找和替换文件中的字符串
使用文本文件时,您经常需要查找并替换一个或多个文件中的文本字符串。
sed
是stream editor。它可以对文件和输入流(例如管道)执行基本的文本操作。使用sed
,您可以搜索、查找和替换、插入和删除单词和行。它支持基本和扩展的正则表达式,使您可以匹配复杂的模式。
在本文中,我们将讨论如何查找字符串并用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
。当提供替换标志时,所有出现的内容都将被替换。INPUTFILE
- 要运行命令的文件的名称。
最好在参数周围加上引号,这样 shell 元字符就不会扩展。
让我们看看如何使用 sed
命令来搜索文件中的文本并将其替换为一些最常用的选项和标志。
出于演示目的,我们将使用以下文件:
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
标志。在下面的示例中,我们同时使用g
和I
标志:
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 教程
。
如果您有任何问题或反馈,请随时发表评论。