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

sometext sometext sometext TEXT_TO_BE_REPLACED sometext sometext sometext

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

This line is removed by the admin.

搜索关键字是text_to_be_replace

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


你需要在前后使用通配符(.*)来替换整行:

sed 's/.*TEXT_TO_BE_REPLACED.*/This line is removed by the admin./'

您可以使用change命令替换整行,并使用-i标志进行适当的更改。例如,使用GNU sed:

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

这和上面那个很相似。

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

我经常使用regex从文件中提取数据,我只是用它来替换文字引号\"与// nothing:-)

cat file.csv | egrep '^\"([0-9]{1,3}\.[0-9]{1,3}\.)' | sed  s/\"//g  | cut -d, -f1 > list.txt

在我的makefile中,我使用这个:

@sed -i '/.*Revision:.*/c\'"`svn info -R main.cpp | awk '/^Rev/'`"'' README.md

PS:不要忘记-i实际上改变了文件中的文本…因此,如果你定义为“Revision”的模式将改变,你也将改变模式来替换。

示例输出:

abc项目由John Doe撰写 修改:1190

因此,如果你设置模式“Revision: 1190”,这显然与你将它们定义为“Revision:”不同……


以上答案:

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

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


bash-4.1$ new_db_host="DB_HOSTNAME=good replaced with 122.334.567.90"
bash-4.1$ 
bash-4.1$ sed -i "/DB_HOST/c $new_db_host" test4sed
vim test4sed
'
'
'
DB_HOSTNAME=good replaced with 122.334.567.90
'

它运行正常


到目前为止提供的所有答案都假设您了解要替换的文本,这是有意义的,因为这是OP所要求的。我提供的答案假设您对要替换的文本一无所知,并且文件中可能有与您不希望被替换的内容相同或类似的单独行。此外,我假设您知道要替换的行号。

下面的例子演示了如何通过特定的行号删除或更改文本:

# replace line 17 with some replacement text and make changes in file (-i switch)
# the "-i" switch indicates that we want to change the file. Leave it out if you'd
#   just like to see the potential changes output to the terminal window.
# "17s" indicates that we're searching line 17
# ".*" indicates that we want to change the text of the entire line
# "REPLACEMENT-TEXT" is the new text to put on that line
# "PATH-TO-FILE" tells us what file to operate on
sed -i '17s/.*/REPLACEMENT-TEXT/' PATH-TO-FILE

# replace specific text on line 3
sed -i '3s/TEXT-TO-REPLACE/REPLACEMENT-TEXT/'

cat find_replace | while read pattern replacement ; do
sed -i "/${pattern}/c ${replacement}" file    
done 

Find_replace文件包含2列,c1为匹配模式,c2为替换,sed循环替换包含变量1的模式之一的每一行


用于配置文件的操作

我受到skensell答案的启发想出了这个解决方案

保密[搜索][重播][文件]

它将:

如果不存在,则创建该文件 替换searchPattern匹配的整行(所有行) 如果没有找到pattern,则在文件末尾添加replaceLine

功能:

function configLine {
  local OLD_LINE_PATTERN=$1; shift
  local NEW_LINE=$1; shift
  local FILE=$1
  local NEW=$(echo "${NEW_LINE}" | sed 's/\//\\\//g')
  touch "${FILE}"
  sed -i '/'"${OLD_LINE_PATTERN}"'/{s/.*/'"${NEW}"'/;h};${x;/./{x;q100};x}' "${FILE}"
  if [[ $? -ne 100 ]] && [[ ${NEW_LINE} != '' ]]
  then
    echo "${NEW_LINE}" >> "${FILE}"
  fi
}

疯狂的退出状态魔术来自https://stackoverflow.com/a/12145797/1262663


下面的命令对我有用。哪个是处理变量的

sed -i "/\<$E\>/c $D" "$B"

要做到这一点,不依赖任何GNUisms,如-i不带参数或c不带换行符:

sed '/TEXT_TO_BE_REPLACED/c\
This line is removed by the admin.
' infile > tmpfile && mv tmpfile infile

在这种(POSIX兼容的)命令形式中

c\
text

文本可以由一行或多行组成,应该成为替换的一部分的换行符必须转义:

c\
line1\
line2
s/x/y/

其中s/x/y/是一个新的sed命令后,模式空间已替换为两行

line1
line2

将包含指定字符串的整行替换为该行的内容

文本文件:

Row: 0 last_time_contacted=0, display_name=Mozart, _id=100, phonebook_bucket_alt=2
Row: 1 last_time_contacted=0, display_name=Bach, _id=101, phonebook_bucket_alt=2

单一的字符串:

$ sed 's/.* display_name=\([[:alpha:]]\+\).*/\1/'
output:
100
101

由空格分隔的多个字符串:

$ sed 's/.* display_name=\([[:alpha:]]\+\).* _id=\([[:digit:]]\+\).*/\1 \2/'
output:
Mozart 100
Bach 101

调整regex以满足您的需要

[:alpha]和[:digit:] 字符类和括号表达式


这招对我很管用:

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的解决方案,但如果这不是一个困难的要求,那么还有另一个可能是更明智的选择的选项。

公认的答案是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,请向系统管理员投诉。