我需要在一个目录中的所有文件和子目录中递归地搜索指定的字符串,并用另一个字符串替换此字符串。

我知道查找它的命令可能是这样的:

grep 'string_to_find' -r ./*

但是如何用另一个字符串替换string_to_find的每个实例呢?


当前回答

我得到了答案。

grep -rl matchstring somedir/ | xargs sed -i 's/string1/string2/g'

其他回答

以下是我的做法:

find /path/to/dir -type f -iname "*filename*" -print0 | xargs -0 sed -i '/searchstring/s/old/new/g'

这将查找所有包含文件名在/路径/到/dir下的文件名的文件,而对于每一个找到的文件,搜索搜索字符串并将旧文件替换为新文件。

但是如果你想忽略查找文件名中有文件名字符串的特定文件,可以简单地执行:

find /path/to/dir -type f -print0 | xargs -0 sed -i '/searchstring/s/old/new/g'

这将执行与上面相同的操作,但是针对/path/to/dir下的所有文件。

我得到了答案。

grep -rl matchstring somedir/ | xargs sed -i 's/string1/string2/g'

在git repo中使用find和sed时要非常小心!如果你不排除二进制文件,你会得到这样的错误:

error: bad index file sha1 signature 
fatal: index file corrupt

要解决这个错误,您需要通过用old_string替换new_string来恢复sed。这将恢复替换的字符串,因此您将回到问题的起点。

搜索字符串并替换它的正确方法是跳过find,而是使用grep来忽略二进制文件:

sed -ri -e "s/old_string/new_string/g" $(grep -Elr --binary-files=without-match "old_string" "/files_dir")

感谢@hobs

另一种选择是在globstar中使用perl。

在你的.bashrc(或任何地方)中启用shop -s globstar允许** glob模式递归匹配所有子目录和文件。

因此使用perl -pXe 's/SEARCH/REPLACE/g' -i **将递归 用replace替换SEARCH。

- x标志告诉perl“禁用所有警告”-这意味着 它不会抱怨目录问题。

globstar还允许您执行sed -i 's/SEARCH/REPLACE/g' **/*之类的操作。如果你想在所有扩展名为.ext的子文件中用replace替换SEARCH。

其他解决方案混合使用正则表达式语法。要使用perl/PCRE模式进行搜索和替换,并且只处理匹配的文件,这工作得很好:

grep -rlIZPi 'match1' | xargs -0r perl -pi -e 's/match2/replace/gi;'

Match1和match2通常是相同的,但match2可以包含更高级的功能,这些功能只与替换相关,例如捕获组。

翻译:grep递归和列表匹配的文件名,每个由null分隔,以保护任何特殊字符;将任何文件名管道到xargs,该xargs需要一个空分隔的列表;如果接收到任何文件名,则将它们传递给perl以执行实际的替换。

对于区分大小写的匹配,从grep中删除i标志,从s///表达式中删除i模式修饰符,但不要从perl本身中删除i标志。要包含二进制文件,请从grep中删除I标志。