查找和替换一个给定的输入字符串,比如abc,并替换成另一个字符串,比如文件/tmp/file.txt中的XYZ,最简单的方法是什么?
我正在编写一个应用程序,并使用IronPython通过SSH执行命令-但我不太了解Unix,也不知道该寻找什么。
我听说Bash除了是一种命令行界面外,还可以是一种非常强大的脚本语言。所以,如果这是真的,我假设你可以执行这样的操作。
我可以用bash实现它吗?要实现我的目标,最简单的(一行)脚本是什么?
查找和替换一个给定的输入字符串,比如abc,并替换成另一个字符串,比如文件/tmp/file.txt中的XYZ,最简单的方法是什么?
我正在编写一个应用程序,并使用IronPython通过SSH执行命令-但我不太了解Unix,也不知道该寻找什么。
我听说Bash除了是一种命令行界面外,还可以是一种非常强大的脚本语言。所以,如果这是真的,我假设你可以执行这样的操作。
我可以用bash实现它吗?要实现我的目标,最简单的(一行)脚本是什么?
文件操作通常不是由Bash完成的,而是由Bash调用的程序完成的,例如:
perl -pi -e 's/abc/XYZ/g' /tmp/file.txt
-i标志告诉它进行就地替换。
有关详细信息,请参阅man perlrun,包括如何对原始文件进行备份。
Bash与其他shell一样,只是一个协调其他命令的工具。通常您会尝试使用标准的UNIX命令,但当然您可以使用Bash调用任何东西,包括您自己编译的程序、其他shell脚本、Python和Perl脚本等。
在这种情况下,有几种方法可以做到这一点。
如果你想读取一个文件,并将其写入另一个文件,执行搜索/替换,使用sed:
sed 's/abc/XYZ/g' <infile >outfile
如果你想就地编辑文件(就像在编辑器中打开文件,编辑它,然后保存它),向行编辑器'ex'提供指令
echo "%s/abc/XYZ/g
w
q
" | ex file
示例就像没有全屏模式的vi。您可以向它输入与vi的:prompt相同的命令。
最简单的方法是使用sed(或perl):
sed -i -e 's/abc/XYZ/g' /tmp/file.txt
由于-i选项,它将调用sed执行就地编辑。这可以从bash调用。
如果你真的只想使用bash,那么下面的方法可以工作:
while IFS='' read -r a; do
echo "${a//abc/XYZ}"
done < /tmp/file.txt > /tmp/file.txt.t
mv /tmp/file.txt{.t,}
这将遍历每行,执行替换,并写入临时文件(不想破坏输入)。最后的移动只是暂时移动到原来的名称。(为了健壮性和安全性,临时文件名不应该是静态的或可预测的,但我们还是不要这么做。)
Mac用户:
sed -i '' 's/abc/XYZ/g' /tmp/file.txt
(原因请看下面的评论)
你也可以使用ed命令进行文件内搜索和替换:
# delete all lines matching foobar
ed -s test.txt <<< $'g/foobar/d\nw'
参见“使用ed通过脚本编辑文件”。
我在其他帖子中发现了这个帖子,我认为它包含了最完整的答案,所以我也添加了我的答案:
sed and ed are so useful...by hand. Look at this code from @Johnny: sed -i -e 's/abc/XYZ/g' /tmp/file.txt When my restriction is to use it in a shell script, no variable can be used inside in place of "abc" or "XYZ". The BashFAQ seems to agree with what I understand at least. So, I can't use: x='abc' y='XYZ' sed -i -e 's/$x/$y/g' /tmp/file.txt #or, sed -i -e "s/$x/$y/g" /tmp/file.txt but, what can we do? As, @Johnny said use a while read... but, unfortunately that's not the end of the story. The following worked well with me: #edit user's virtual domain result= #if nullglob is set then, unset it temporarily is_nullglob=$( shopt -s | egrep -i '*nullglob' ) if [[ is_nullglob ]]; then shopt -u nullglob fi while IFS= read -r line; do line="${line//'<servername>'/$server}" line="${line//'<serveralias>'/$alias}" line="${line//'<user>'/$user}" line="${line//'<group>'/$group}" result="$result""$line"'\n' done < $tmp echo -e $result > $tmp #if nullglob was set then, re-enable it if [[ is_nullglob ]]; then shopt -s nullglob fi #move user's virtual domain to Apache 2 domain directory ...... As one can see if nullglob is set then, it behaves strangely when there is a string containing a * as in: <VirtualHost *:80> ServerName www.example.com which becomes <VirtualHost ServerName www.example.com there is no ending angle bracket and Apache2 can't even load. This kind of parsing should be slower than one-hit search and replace but, as you already saw, there are four variables for four different search patterns working out of one parse cycle.
在给定的假设下,我能想到的最合适的解决方案。
当我被这个绊倒时,我很惊讶……
“mysql-server”包中有一个replace命令,如果你已经安装了它,请尝试一下:
# replace string abc to XYZ in files
replace "abc" "XYZ" -- file.txt file2.txt file3.txt
# or pipe an echo to replace
echo "abcdef" |replace "abc" "XYZ"
关于这一点,请参阅man replace。
可以使用rpl命令。例如,你想改变整个php项目的域名。
rpl -ivRpd -x'.php' 'old.domain.name' 'new.domain.name' ./path_to_your_project_folder/
这不是一个明确的bash的原因,但它是一个非常快速和有用的。:)
要以非交互式方式编辑文件中的文本,需要就地文本编辑器,例如vim。
下面是一个简单的例子,如何从命令行使用它:
vim -esnc '%s/foo/bar/g|:wq' file.txt
这相当于ex editor的@slim answer,基本上是一样的。
这里有一些实际的例子。
在文件中用bar替换文本foo:
ex -s +%s/foo/bar/ge -cwq file.txt
删除多个文件的尾随空格:
ex +'bufdo!%s/\s\+$//e' -cxa *.txt
故障处理(终端卡死):
添加-V1参数以显示详细消息。 强制退出:-cwq!
参见:
如何非交互地编辑文件(例如在管道中)?在Vi SE
使用“/”字符替换url时要小心。
如何做到这一点的例子:
sed -i "s%http://domain.com%http://www.domain.com/folder/%g" "test.txt"
摘自:http://www.sysadmit.com/2015/07/linux-reemplazar-texto-en-archivos-con-sed.html
这是一篇旧文章,但对于任何想要使用变量的人来说,@centurian说单引号意味着不会展开任何内容。
获取变量的一个简单方法是做字符串拼接,因为这是通过在bash中并置完成的,以下应该工作:
sed -i -e "s/$var1/$var2/g" /tmp/file.txt
您也可以在bash脚本中使用python。我在这里的一些顶级答案上没有太大的成功,并且发现这个不需要循环就能工作:
#!/bin/bash
python
filetosearch = '/home/ubuntu/ip_table.txt'
texttoreplace = 'tcp443'
texttoinsert = 'udp1194'
s = open(filetosearch).read()
s = s.replace(texttoreplace, texttoinsert)
f = open(filetosearch, 'w')
f.write(s)
f.close()
quit()
你可以使用sed:
sed -i 's/abc/XYZ/gi' /tmp/file.txt
如果你不知道你的文件名,你可以使用find和sed:
find ./ -type f -exec sed -i 's/abc/XYZ/gi' {} \;
在所有Python文件中查找并替换:
find ./ -iname "*.py" -type f -exec sed -i 's/abc/XYZ/gi' {} \;
如果你正在处理的文件不是很大,并且暂时将它存储在一个变量中是没有问题的,那么你可以在整个文件上使用Bash字符串替换—没有必要逐行检查它:
file_contents=$(</tmp/file.txt)
echo "${file_contents//abc/XYZ}" > /tmp/file.txt
整个文件内容将被视为一个长字符串,包括换行符。
XYZ可以是一个变量,例如$replacement,在这里不使用sed的一个好处是,您不必担心搜索或替换字符串可能包含sed模式分隔符(通常,但不一定是/)。缺点是不能使用正则表达式或sed的任何更复杂的操作。
使用sed命令替换文件中多个文本的最简单方法
命令- - - - - -
sed -i s#a/b/c#D/E#g;s#/x/y/z#D:/ x #g
在上面的命令s#a/b/c#D/E#g中,我用D/E替换a/b/c,然后在;我们又在做同样的事情
对于MAC用户,如果你没有读评论的话:)
正如@Austin提到的,如果您得到无效的命令代码错误
对于就地替换,BSD sed要求在-i标志之后添加一个文件扩展名,以保存到具有给定扩展名的备份文件中。
Sed -i '.bak'的/find/replace' /file.txt
如果您想跳过备份,可以使用“空字符串”。
Sed -i " 's/find/replace' /file.txt
这都是奥斯汀的功劳
如果在多个文件中一起做更改,我们可以在一行中做:-
user_name='whoami'
for file in file1.txt file2.txt file3.txt; do sed -i -e 's/default_user/${user_name}/g' $file; done
补充如果万一可能有用。