I have "I love Suzi and Marry" and I want to change "Suzi" to "Sara".
firstString="I love Suzi and Marry"
secondString="Sara"
Desired result:
firstString="I love Sara and Marry"
I have "I love Suzi and Marry" and I want to change "Suzi" to "Sara".
firstString="I love Suzi and Marry"
secondString="Sara"
Desired result:
firstString="I love Sara and Marry"
To replace the first occurrence of a pattern with a given string, use ${parameter/pattern/string}
:
#!/bin/bash
firstString="I love Suzi and Marry"
secondString="Sara"
echo "${firstString/Suzi/"$secondString"}"
# prints 'I love Sara and Marry'
To replace all occurrences, use ${parameter//pattern/string}
:
message='The secret code is 12345'
echo "${message//[0-9]/X}"
# prints 'The secret code is XXXXX'
(This is documented in the Bash Reference Manual, §3.5.3 "Shell Parameter Expansion".)
Note that this feature is not specified by POSIX — it's a Bash extension — so not all Unix shells implement it. For the relevant POSIX documentation, see The Open Group Technical Standard Base Specifications, Issue 7, the Shell & Utilities volume, §2.6.2 "Parameter Expansion".
$STRING="${STRING/\n/<br />}"
–
Mutton \n
in that context would represent itself, not a newline. I don't have Bash handy right now to test, but you should be able to write something like, $STRING="${STRING/$'\n'/<br />}"
. (Though you probably want STRING//
-- replace-all -- instead of just STRING/
.) –
Bridegroom foo=/home/user/dir
, followed by echo "${foo/\/home\/user//home}"
, will print /home/dir
. –
Bridegroom echo ${my string foo/foo/bar}
. You'd need input="my string foo"; echo ${input/foo/bar}
–
Sternway Marry
, end of story. –
Bridegroom foo = "$(pwd)"
and bar="/test/"
how can I use now echo "${foo/bar/profi/}"
–
Pentheam path=/home/me/docs; echo "${path/me\/docs/you\/docs}"
–
Chicanery ${${I like Suzi and Marry/Suzi/Sara}/like/love}
, ${I like Suzi and Marry/Suzi/Sara/like/love}
, or something else? –
Herthahertz //
explained? And which characters do I need to escape in match and replace ? –
Pester //
directly, it mentions what happens "If pattern begins with ‘/
’" (which isn't even accurate, since the rest of that sentence assumes that the extra /
is not actually part of the pattern) -- but I don't know of any better source. –
Bridegroom This can be done entirely with Bash string manipulation:
first="I love Suzy and Mary"
second="Sara"
first=${first/Suzy/$second}
That will replace only the first occurrence; to replace them all, double the first slash:
first="Suzy, Suzy, Suzy"
second="Sara"
first=${first//Suzy/$second}
# first is now "Sara, Sara, Sara"
first
or second
contain special characters, like /
, $
, {
, }
, <backslash>, .
, +
, (
, )
, *
, etc.? (Problems with formatting of backslash in this comment.) Perhaps address that in the answer? –
Dukes r_getfilter="${r_getfilter/../.*}"
to replace ..
with .*
(a grep "match anything" string). Using '.*'
did not go well, the single quotes ended up in the substitution. –
Fellmonger For Dash all previous posts aren't working
The POSIX sh
compatible solution is:
result=$(echo "$firstString" | sed "s/Suzi/$secondString/")
This will replace the first occurrence on each line of input. Add a /g
flag to replace all occurrences:
result=$(echo "$firstString" | sed "s/Suzi/$secondString/g")
result=$(echo $firstString | sed "s/Suzi/$secondString/g")
–
Palace echo
argument. It deceptively works without quoting with simple strings, but easily breaks on any nontrivial input string (irregular spacing, shell metacharacters, etc). –
Stucker Suzi
) is a regular expression, so some characters have special meaning (try matching a dot .
) –
Unfledged Try this:
sed "s/Suzi/$secondString/g" <<< "$firstString"
The three greater-than signs create a here string.
zsh: command too long
. –
Dietetic my_command | sed ... | ...
–
Quaggy <<<
"here string" syntax is not POSIX. –
Stucker It's better to use Bash than sed
if strings have regular expression characters.
echo ${first_string/Suzi/$second_string}
It's portable to Windows and works with at least as old as Bash 3.1.
To show you don't need to worry much about escaping, let's turn this:
/home/name/foo/bar
Into this:
~/foo/bar
But only if /home/name
is in the beginning. We don't need sed
!
Given that Bash gives us magic variables $PWD
and $HOME
, we can:
echo "${PWD/#$HOME/\~}"
Thanks for Mark Haferkamp in the comments for the note on quoting/escaping ~
.*
Note how the variable $HOME
contains slashes, but this didn't break anything.
Further reading: Advanced Bash-Scripting Guide.
If using sed
is a must, be sure to escape every character.
sed
with the pwd
command to avoid defining a new variable each time my custom $PS1
runs. Does Bash provide a more general way than magic variables to use the output of a command for string replacement? As for your code, I had to escape the ~
to keep Bash from expanding it into $HOME. Also, what does the #
in your command do? –
Nonresident ~
": notice how I quoted stuff. Remember to always quote stuff! And this doesn't just work for magic variables: any variable is capable of substitutions, getting string length, and more, within bash. Congrats on trying to your $PS1
fast: you may also be interested in $PROMPT_COMMAND
if you are more comfortable in another programming language and want to code a compiled prompt. –
Oscillatory echo "${PWD/#$HOME/~}"
doesn't replace my $HOME
with ~
. Replacing ~
with \~
or '~'
works. Any of these work on Bash 4.2.53 on another distro. Can you please update your post to quote or escape the ~
for better compatibility? What I meant by my "magic variables" question was: Can I use Bash's variable substitution on, e.g., the output of uname
without saving it as a variable first? As for my personal $PROMPT_COMMAND
, it's complicated. –
Nonresident ${PWD/#$HOME/~}
works in Bash 4.2.53 on SUSE and ${PWD/#$HOME/\~}
works in Bash 4.3.30 on Chakra, but not vice versa. A workaround that works on both distros is to replace ~
with $(echo '~')
, as in the verbose ${PWD/#$HOME/$(echo '~')}
. Edit: I just realized that $'~'
might be better. I'm testing it now. –
Nonresident $'~'
works great in an echo command, but caused me nothing but frustration when I tried to use it in $PS1 to get a literal ~
. $(echo '~')
seems to be the best workaround. –
Nonresident $HOME/
but you'll have to escape the slash –
Gisele echo [string] | sed "s|[original]|[target]|g"
g
flag is hugely misunderstood. Without it, sed
will replace the first occurrence on each line but if you don't expect multiple occurrences per line, you don't need g
. (Frequently you see it in expressions where there could only ever be a single match per line, like s/.*everything.*/all of it/g
where obviously you are matching the entire line in the first place, so there is no way you could match the regex several times). –
Stucker If tomorrow you decide you don't love Marry either she can be replaced as well:
today=$(</tmp/lovers.txt)
tomorrow="${today//Suzi/Sara}"
echo "${tomorrow//Marry/Jesica}" > /tmp/lovers.txt
There must be 50 ways to leave your lover.
Since I can't add a comment. @ruaka To make the example more readable write it like this
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}
Pure POSIX shell method, which unlike Roman Kazanovskyi's sed
-based answer needs no external tools, just the shell's own native parameter expansions. Note that long file names are minimized so the code fits better on one line:
f="I love Suzi and Marry"
s=Sara
t=Suzi
[ "${f%$t*}" != "$f" ] && f="${f%$t*}$s${f#*$t}"
echo "$f"
Output:
I love Sara and Marry
How it works:
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.
[ "${f%"$t"*}" != "$f" ] && f="${f%"$t"*}$s${f#*"$t"}"]
–
Branchiopod Using AWK:
firstString="I love Suzi and Marry"
echo "$firstString" | awk '{gsub("Suzi","Sara"); print}'
Pattern to substitute the first occurrence with special charters:
${parameter/pattern/string}
Pattern to substitute all occurrence with special charters:
${parameter//pattern/string}
firstString="I love //Suzi// and Marry"
secondString="Sara"
firstString="${firstString/\/\/Suzi\/\//"$secondString"}"
echo $firstString
It will print: I love Sara and Marry
I think this is the cleanest form for your use case:
firstString="${firstString//Suzi/$secondString}"
Try this:
ls *.ext | awk '{print "mv "$1" "$1".newext"}' | sed "s/.ext.newext/.newext/" | parallel {}
based on proposed above awk solution, I would extend it to use awk-variables. This will allow passing a text containing special chars..
aString="I love _p1_ very much!"
aVar="complicated \" text \' with \. special ) chars"
awk -v p1="$aVar" '{gsub("_p1_",p1); print}' <<< $aString
produces:
I love complicated " text ' with . special ) chars very much
it would be uneasy to implement this case with sed -e
or bash
substitutions.
As python now builtin available in linux, I would suggest this py string replace str.replace()
firstString="I love Suzi and Marry"
secondString="Sara"
secondString=`python3 -c "s='$firstString'.replace('Suzi', 'Sara'); print(s)" `
echo $secondString
The only way I found is store the string in a file, use sed then store the file content in a var :
echo "I love Suzy" > tmp.txt
sed -i "s/Suzy/Sarah/" tmp.txt
set res=`cat tmp.txt`
echo $res
rm tmp.txt
I don't know which kind of shell I am using (only thing I found is sh-4.2 if I type 'sh') but all classic syntax fails, like the simple test=${test2}
.
It fails 2 times : at the assignment (must use set
) and at the ${}
.
csh
, but even that supported pipes. sh-4.2
looks like Bash running in POSIX compatibility mode, but that would not require (or understand) the set
syntax. There was a time when Csh was quite popular in spite of its obvious flaws, but that was 30+ years ago; probably join us in abandoning it. –
Stucker Using sed
we can do it easily
sed -i "s+$value_to_be_replaced+$with_variable1 "some character" $with_variable2+g" $file_name
© 2022 - 2024 — McMap. All rights reserved.