我有“我爱Suzi和Marry”,我想把“Suzi”改成“Sara”。
firstString="I love Suzi and Marry"
secondString="Sara"
预期的结果:
firstString="I love Sara and Marry"
我有“我爱Suzi和Marry”,我想把“Suzi”改成“Sara”。
firstString="I love Suzi and Marry"
secondString="Sara"
预期的结果:
firstString="I love Sara and Marry"
使用${parameter/pattern/string}替换模式的第一次出现:
#!/bin/bash
firstString="I love Suzi and Marry"
secondString="Sara"
echo "${firstString/Suzi/"$secondString"}"
# prints 'I love Sara and Marry'
使用${parameter//pattern/string}替换所有出现的情况:
message='The secret code is 12345'
echo "${message//[0-9]/X}"
# prints 'The secret code is XXXXX'
(这在Bash参考手册§3.5.3“Shell参数扩展”中有记录。)
注意这个特性不是POSIX指定的——它是一个Bash扩展——所以不是所有Unix shell都实现它。有关POSIX的相关文档,请参见Open Group技术标准基础规范,第7期,Shell & Utilities卷,§2.6.2“参数扩展”。
这完全可以用bash字符串操作完成:
first="I love Suzy and Mary"
second="Sara"
first=${first/Suzy/$second}
这只会取代第一个事件;要将它们全部替换,请将第一个斜杠加倍:
first="Suzy, Suzy, Suzy"
second="Sara"
first=${first//Suzy/$second}
# first is now "Sara, Sara, Sara"
对于Dash,之前的所有帖子都不起作用
POSIX sh兼容的解决方案是:
result=$(echo "$firstString" | sed "s/Suzi/$secondString/")
这将替换每一行输入中的第一个出现的位置。添加/g标记替换所有出现的情况:
result=$(echo "$firstString" | sed "s/Suzi/$secondString/g")
如果字符串包含正则表达式字符,那么使用Bash比使用sed更好。
echo ${first_string/Suzi/$second_string}
它可以移植到Windows,并且至少可以与Bash 3.1兼容。
为了表明你不需要太担心逃跑,让我们把这个:
/home/name/foo/bar
到这个:
~/foo/bar
但前提是/home/name在开头。我们不需要sed!
假设Bash给了我们神奇的变量$PWD和$HOME,我们可以:
echo "${PWD/#$HOME/\~}"
感谢Mark Haferkamp在评论中提到引用/转义~.*
注意变量$HOME包含斜杠,但这并没有破坏任何东西。
进一步阅读:高级bash脚本编写指南。 如果必须使用sed,请确保转义每个字符。
如果明天你决定不再爱她,她也可以被取代:
today=$(</tmp/lovers.txt)
tomorrow="${today//Suzi/Sara}"
echo "${tomorrow//Marry/Jesica}" > /tmp/lovers.txt
离开爱人,一定有50种方法。
因为我不能添加评论。为了使示例更具可读性,可以这样写
full_string="I love Suzy and Mary"
search_string="Suzy"
replace_string="Sara"
my_string=${full_string/$search_string/$replace_string}
or
my_string=${full_string/Suzy/Sarah}
纯POSIX shell方法,与Roman Kazanovskyi基于sed的答案不同,它不需要外部工具,只需要shell自己的本地参数展开。注意,长文件名被最小化,所以代码更适合一行:
f="I love Suzi and Marry"
s=Sara
t=Suzi
[ "${f%$t*}" != "$f" ] && f="${f%$t*}$s${f#*$t}"
echo "$f"
输出:
I love Sara and Marry
工作原理:
Remove Smallest Suffix Pattern. "${f%$t*}" returns "I love" if the suffix $t "Suzi*" is in $f "I love Suzi and Marry". But if t=Zelda, then "${f%$t*}" deletes nothing, and returns the whole string "I love Suzi and Marry". This is used to test if $t is in $f with [ "${f%$t*}" != "$f" ] which will evaluate to true if the $f string contains "Suzi*" and false if not. If the test returns true, construct the desired string using Remove Smallest Suffix Pattern ${f%$t*} "I love" and Remove Smallest Prefix Pattern ${f#*$t} "and Marry", with the 2nd string $s "Sara" in between.
使用AWK:
firstString="I love Suzi and Marry"
echo "$firstString" | awk '{gsub("Suzi","Sara"); print}'
我发现的唯一方法是将字符串存储在一个文件中,使用sed,然后将文件内容存储在一个var中:
echo "I love Suzy" > tmp.txt
sed -i "s/Suzy/Sarah/" tmp.txt
set res=`cat tmp.txt`
echo $res
rm tmp.txt
我不知道我正在使用哪种shell(如果我键入'sh',我唯一发现的是sh-4.2),但所有经典语法都失败了,比如简单的test=${test2}。 它失败了2次:在赋值处(必须使用set)和${}处。
试试这个:
ls *.ext | awk '{print "mv "$1" "$1".newext"}' | sed "s/.ext.newext/.newext/" | parallel {}
模式,将第一次出现替换为特殊的包机:
${参数/模式/字符串}
将所有发生的情况替换为特殊特许状的模式:
${参数/ /模式/字符串}
firstString="I love //Suzi// and Marry"
secondString="Sara"
firstString="${firstString/\/\/Suzi\/\//"$secondString"}"
echo $firstString
上面印着:我爱莎拉和玛丽
使用sed我们可以很容易地做到这一点
sed -i "s+$value_to_be_replaced+$with_variable1 "some character" $with_variable2+g" $file_name