我有一个文本文件,其中有一行

sometext sometext sometext TEXT_TO_BE_REPLACED sometext sometext sometext

我需要将上面的整行替换为

This line is removed by the admin.

搜索关键字是text_to_be_replace

我需要为此写一个shell脚本。如何使用sed实现这一点?


当前回答

以上答案:

sed -i '/TEXT_TO_BE_REPLACED/c\This line is removed by the admin.' /tmp/foo

如果替换字符串/行不是变量,工作正常。

问题是在Redhat 5上,c后面的\转义了$。双\\也不起作用(至少在红帽5上)。

通过hit和trial,我发现如果替换字符串/行只有一行,那么c后面的\是多余的。所以我没有在c后面使用\,使用一个变量作为单一的替换行,它是joy。

代码看起来像这样:

sed -i "/TEXT_TO_BE_REPLACED/c $REPLACEMENT_TEXT_STRING" /tmp/foo

注意使用双引号而不是单引号。

其他回答

这招对我很管用:

sed -i <extension> 's/.*<Line to be replaced>.*/<New line to be added>/'

一个例子是:

sed -i .bak -e '7s/.*version.*/      version = "4.33.0"/'

-i:替换后备份文件的扩展名。在本例中,它是.bak。 -e: sed脚本。在本例中,它是'7s/.*版本。*/ version = "4.33.0"/'。如果您想使用sed文件,请使用-f标志 s:要替换的文件中的行号。在本例中,它是7s,也就是第7行。

注意:

如果你想做一个递归查找和替换sed,那么你可以grep到命令的开头:

grep -rl --exclude-dir=<directory-to-exclude> --include=\*<Files to include> "<Line to be replaced>" ./ | sed -i <extension> 's/.*<Line to be replaced>.*/<New line to be added>/'

这和上面那个很相似。

sed 's/[A-Za-z0-9]*TEXT_TO_BE_REPLACED.[A-Za-z0-9]*/This line is removed by the admin./'

这个公认的答案并不适合我,原因如下:

我的sed版本不喜欢使用0长度扩展的-i c\命令的语法很奇怪,我不能让它工作 我没有意识到我的一些问题来自于未转义的斜杠

以下是我提出的解决方案,我认为它适用于大多数情况:

function escape_slashes {
    sed 's/\//\\\//g' 
}

function change_line {
    local OLD_LINE_PATTERN=$1; shift
    local NEW_LINE=$1; shift
    local FILE=$1

    local NEW=$(echo "${NEW_LINE}" | escape_slashes)
    # FIX: No space after the option i.
    sed -i.bak '/'"${OLD_LINE_PATTERN}"'/s/.*/'"${NEW}"'/' "${FILE}"
    mv "${FILE}.bak" /tmp/
}

因此本例使用提出了修正问题的方法:

change_line "TEXT_TO_BE_REPLACED" "This line is removed by the admin." yourFile

以上答案:

sed -i '/TEXT_TO_BE_REPLACED/c\This line is removed by the admin.' /tmp/foo

如果替换字符串/行不是变量,工作正常。

问题是在Redhat 5上,c后面的\转义了$。双\\也不起作用(至少在红帽5上)。

通过hit和trial,我发现如果替换字符串/行只有一行,那么c后面的\是多余的。所以我没有在c后面使用\,使用一个变量作为单一的替换行,它是joy。

代码看起来像这样:

sed -i "/TEXT_TO_BE_REPLACED/c $REPLACEMENT_TEXT_STRING" /tmp/foo

注意使用双引号而不是单引号。

问题要求使用sed的解决方案,但如果这不是一个困难的要求,那么还有另一个可能是更明智的选择的选项。

公认的答案是sed -i,并将其描述为替换文件的位置,但-i并没有真正这样做,而是执行等效的sed模式文件> tmp;Mv TMP文件,保留所有权和模式。这在很多情况下并不理想。一般来说,我不建议以非交互式的方式运行sed -i作为自动过程的一部分——这就像设置了一个引信长度未知的炸弹。它迟早会在某人身上爆炸。

要真正“就地”编辑一个文件,用其他内容替换与模式匹配的行,您可以使用实际的文本编辑器。这就是标准文本编辑器ed的工作方式。

printf '%s\n' '/TEXT_TO_BE_REPLACED/' d i 'This line is removed by the admin' . w q | \
ed -s /tmp/foo > /dev/null

注意,这只替换第一个匹配的行,这是问题所暗示的需要的。这与大多数其他答案有本质区别。

撇开这个缺点不谈,ed比sed有一些优点:

您可以用一行或多行替换匹配,而不需要任何额外的工作。 替换文本可以任意复杂,而不需要任何转义来保护它。 最重要的是,原始文件将被打开、修改和保存。没有复制。

它是如何工作的

工作原理:

printf will use its first argument as a format string and print each of its other arguments using that format, effectively meaning that each argument to printf becomes a line of output, which is all sent to ed on stdin. The first line is a regex pattern match which causes ed to move its notion of "the current line" forward to the first line that matches (if there is no match the current line is set to the last line of the file). The next is the d command which instructs ed to delete the entire current line. After that is the i command which puts ed into insert mode; after that all subsequent lines entered are written to the current line (or additional lines if there are any embedded newlines). This means you can expand a variable (e.g. "$foo") containing multiple lines here and it will insert all of them. Insert mode ends when ed sees a line consisting of . The w command writes the content of the file to disk, and the q command quits. The ed command is given the -s switch, putting it into silent mode so it doesn't echo any information as it runs, the file to be edited is given as an argument to ed, and, finally, stdout is thrown away to prevent the line matching the regex from being printed.

一些类unix系统可能(不恰当地)在没有安装ed的情况下发布,但仍然可能附带ex;如果是这样,你可以简单地用它来代替。如果有vim但没有ex或ed,可以使用vim -e代替。如果您只有标准的vi,而没有ex或ed,请向系统管理员投诉。