big-z
big-z

Reputation: 7102

How to replace all occurrences of a character in string?

What is the effective way to replace all occurrences of a character with another character in std::string?

Upvotes: 647

Views: 778151

Answers (19)

Amit
Amit

Reputation: 1160

A single line of code in standard C++:

C++20

#include <string>
#include <algorithm>
#include <iostream>

int main()
{
    std::string str{ "This is a test string." };

    std::ranges::replace(str, 'i', 'p'); // -> "Thps ps a test strpng."

    std::cout << str << std::endl;
}

Demo


Prior to C++20

#include <string>
#include <algorithm>
#include <iostream>

int main()
{
    std::string str{ "This is a test string." };

    std::replace(str.begin(), str.end(), 'i', 'p'); // -> "Thps ps a test strpng."

    std::cout << str << std::endl;
}

Demo

Upvotes: 5

Louis Go
Louis Go

Reputation: 2588

If you don't know whether to use the answers std::range::replace or std::replace

Here's the benchmark for replacing a string length ~300 long string.
Quick benchmark Both of GCC and Clang have similar results.

On GCC 13.2 with -O3, they are nearly the same (std::replace: 1565.638, std::range::replace: 1557.231

On Clang 16.0 with -O3 std::replace: 395.697, std::range::replace: 404.269

Source code:

constexpr static std::string_view TEST_STRING = "aslkdfjl;aksjdflk;gajslkdfjaslk;dfjlkasd jflkjlkjlkjsldfka aslkdfjl;aksjdflk;gajslkdfjaslk;dfjlkasd jflkjlkjlkjsldfka aslkdfjl;aksjdflk;gajslkdfjaslk;dfjlkasd jflkjlkjlkjsldfka aslkdfjl;aksjdflk;gajslkdfjaslk;dfjlkasd jflkjlkjlkjsldfka aslkdfjl;aksjdflk;gajslkdfjaslk;dfjlkasd jflkjlkjlkjsldfka ";
const char FIND = 'l';
const char REPLACED = 'q';

static void Replace(benchmark::State& state) {
  // Code inside this loop is measured repeatedly
  for (auto _ : state) {
    std::string s(TEST_STRING);
    std::replace(std::begin(s), std::end(s),FIND, REPLACED);

    // Make sure the variable is not optimized away by compiler
    benchmark::DoNotOptimize(s);
  }
}
// Register the function as a benchmark
BENCHMARK(Replace);

static void RangeReplace(benchmark::State& state) {
  // Code before the loop is not measured
  for (auto _ : state) {
    std::string s(TEST_STRING);
    std::ranges::replace(s,FIND,REPLACED);

    // Make sure the variable is not optimized away by compiler
    benchmark::DoNotOptimize(s);
  }
}
BENCHMARK(RangeReplace);


static void Baseline(benchmark::State& state) {
  // Code before the loop is not measured
  for (auto _ : state) {
    std::string s(TEST_STRING);
    // Make sure the variable is not optimized away by compiler
    benchmark::DoNotOptimize(s);
  }
}
BENCHMARK(Baseline);

Side notes: If you have a long string larger than L3 cache and need to replace with a mapping table (Eg: a->1, b->2...etc) It's better to optimize temporal locality.

Eg: https://quick-bench.com/q/-x3IUSk9YYL-fh0NIFSS6teT9wQ

Upvotes: 0

Gauthier Boaglio
Gauthier Boaglio

Reputation: 10262

The question is centered on character replacement, but, as I found this page very useful (especially Konrad's remark), I'd like to share this more generalized implementation, which allows to deal with substrings as well:

std::string ReplaceAll(std::string str, const std::string& from, const std::string& to) {
    size_t start_pos = 0;
    while((start_pos = str.find(from, start_pos)) != std::string::npos) {
        str.replace(start_pos, from.length(), to);
        start_pos += to.length(); // Handles case where 'to' is a substring of 'from'
    }
    return str;
}

Usage:

std::cout << ReplaceAll(string("Number Of Beans"), std::string(" "), std::string("_")) << std::endl;
std::cout << ReplaceAll(string("ghghjghugtghty"), std::string("gh"), std::string("X")) << std::endl;
std::cout << ReplaceAll(string("ghghjghugtghty"), std::string("gh"), std::string("h")) << std::endl;

Outputs:

Number_Of_Beans

XXjXugtXty

hhjhugthty


EDIT:

The above can be implemented in a more suitable way, in case performance is of your concern, by returning nothing (void) and performing the changes "in-place"; that is, by directly modifying the string argument str, passed by reference instead of by value. This would avoid an extra costly copy of the original string by overwriting it.

Code :

static inline void ReplaceAll2(std::string &str, const std::string& from, const std::string& to)
{
    // Same inner code...
    // No return statement
}

Hope this will be helpful for some others...

Upvotes: 176

chara
chara

Reputation: 121

How about replace any character string with any character string using only good-old C string functions?

char original[256]="First Line\nNext Line\n", dest[256]="";
char* replace_this = "\n"; // this is now a single character but could be any string
char* with_this = "\r\n"; // this is 2 characters but could be of any length

/* get the first token */
char* token = strtok(original, replace_this);

/* walk through other tokens */
while (token != NULL) {
    strcat(dest, token);
    strcat(dest, with_this);
    token = strtok(NULL, replace_this);
}

dest should now have what we are looking for.

Upvotes: 1

emvee
emvee

Reputation: 4449

I think I'd use std::replace_if()

A simple character-replacer (requested by OP) can be written by using standard library functions.

For an in-place version:

#include <string>
#include <algorithm>

void replace_char(std::string& in,
                  std::string::value_type srch,
                  std::string::value_type repl)
{
    std::replace_if(std::begin(in), std::end(in),
                    [&srch](std::string::value_type v) { return v==srch; },
                    repl);
    return;
}

and an overload that returns a copy if the input is a const string:

std::string replace_char(std::string const& in,
                         std::string::value_type srch,
                         std::string::value_type repl)
{
    std::string result{ in };
    replace_char(result, srch, repl);
    return result;
}

Upvotes: 0

Kiruahxh
Kiruahxh

Reputation: 2075

This is not the only method missing from the standard library, it was intended be low level. This use case and many other are covered by general libraries such as:

QtCore & QString has my preference: it supports UTF8 and uses less templates, which means understandable errors and faster compilation. It uses the "q" prefix which makes namespaces unnecessary and simplifies headers.
Boost often generates hideous error messages and slow compile time.
POCO seems to be a reasonable compromise.

Upvotes: 0

Adri&#224; Arrufat
Adri&#224; Arrufat

Reputation: 471

For completeness, here's how to do it with std::regex.

#include <regex>
#include <string>

int main()
{
    const std::string s = "example string";
    const std::string r = std::regex_replace(s, std::regex("x"), "y");
}

Upvotes: 21

alex
alex

Reputation: 955

here's a solution i rolled, in a maximal DRI spirit. it will search sNeedle in sHaystack and replace it by sReplace, nTimes if non 0, else all the sNeedle occurences. it will not search again in the replaced text.

std::string str_replace(
    std::string sHaystack, std::string sNeedle, std::string sReplace, 
    size_t nTimes=0)
{
    size_t found = 0, pos = 0, c = 0;
    size_t len = sNeedle.size();
    size_t replen = sReplace.size();
    std::string input(sHaystack);

    do {
        found = input.find(sNeedle, pos);
        if (found == std::string::npos) {
            break;
        }
        input.replace(found, len, sReplace);
        pos = found + replen;
        ++c;
    } while(!nTimes || c < nTimes);

    return input;
}

Upvotes: 1

hotblack944
hotblack944

Reputation: 181

What about Abseil StrReplaceAll? From the header file:

// This file defines `absl::StrReplaceAll()`, a general-purpose string
// replacement function designed for large, arbitrary text substitutions,
// especially on strings which you are receiving from some other system for
// further processing (e.g. processing regular expressions, escaping HTML
// entities, etc.). `StrReplaceAll` is designed to be efficient even when only
// one substitution is being performed, or when substitution is rare.
//
// If the string being modified is known at compile-time, and the substitutions
// vary, `absl::Substitute()` may be a better choice.
//
// Example:
//
// std::string html_escaped = absl::StrReplaceAll(user_input, {
//                                                {"&", "&amp;"},
//                                                {"<", "&lt;"},
//                                                {">", "&gt;"},
//                                                {"\"", "&quot;"},
//                                                {"'", "&#39;"}});

Upvotes: 4

Old School :-)

std::string str = "H:/recursos/audio/youtube/libre/falta/"; 

for (int i = 0; i < str.size(); i++) {
    if (str[i] == '/') {
        str[i] = '\\';
    }
}

std::cout << str;

Result:

H:\recursos\audio\youtube\libre\falta\

Upvotes: 4

Guney Ozsan
Guney Ozsan

Reputation: 330

For simple situations this works pretty well without using any other library then std::string (which is already in use).

Replace all occurences of character a with character b in some_string:

for (size_t i = 0; i < some_string.size(); ++i) {
    if (some_string[i] == 'a') {
        some_string.replace(i, 1, "b");
    }
}

If the string is large or multiple calls to replace is an issue, you can apply the technique mentioned in this answer: https://stackoverflow.com/a/29752943/3622300

Upvotes: 1

Ingmar
Ingmar

Reputation: 2658

Imagine a large binary blob where all 0x00 bytes shall be replaced by "\1\x30" and all 0x01 bytes by "\1\x31" because the transport protocol allows no \0-bytes.

In cases where:

  • the replacing and the to-replaced string have different lengths,
  • there are many occurences of the to-replaced string within the source string and
  • the source string is large,

the provided solutions cannot be applied (because they replace only single characters) or have a performance problem, because they would call string::replace several times which generates copies of the size of the blob over and over. (I do not know the boost solution, maybe it is OK from that perspective)

This one walks along all occurrences in the source string and builds the new string piece by piece once:

void replaceAll(std::string& source, const std::string& from, const std::string& to)
{
    std::string newString;
    newString.reserve(source.length());  // avoids a few memory allocations

    std::string::size_type lastPos = 0;
    std::string::size_type findPos;

    while(std::string::npos != (findPos = source.find(from, lastPos)))
    {
        newString.append(source, lastPos, findPos - lastPos);
        newString += to;
        lastPos = findPos + from.length();
    }

    // Care for the rest after last occurrence
    newString += source.substr(lastPos);

    source.swap(newString);
}

Upvotes: 51

oOpSgEo
oOpSgEo

Reputation: 193

This works! I used something similar to this for a bookstore app, where the inventory was stored in a CSV (like a .dat file). But in the case of a single char, meaning the replacer is only a single char, e.g.'|', it must be in double quotes "|" in order not to throw an invalid conversion const char.

#include <iostream>
#include <string>

using namespace std;

int main()
{
    int count = 0;  // for the number of occurences.
    // final hold variable of corrected word up to the npos=j
    string holdWord = "";
    // a temp var in order to replace 0 to new npos
    string holdTemp = "";
    // a csv for a an entry in a book store
    string holdLetter = "Big Java 7th Ed,Horstman,978-1118431115,99.85";

    // j = npos
    for (int j = 0; j < holdLetter.length(); j++) {

        if (holdLetter[j] == ',') {

            if ( count == 0 ) 
            {           
                holdWord = holdLetter.replace(j, 1, " | ");      
            }
            else {

                string holdTemp1 = holdLetter.replace(j, 1, " | ");

                // since replacement is three positions in length,
                // must replace new replacement's 0 to npos-3, with
                // the 0 to npos - 3 of the old replacement 
                holdTemp = holdTemp1.replace(0, j-3, holdWord, 0, j-3); 

                holdWord = "";

                holdWord = holdTemp;

            }
            holdTemp = "";
            count++;
        }
    } 
    cout << holdWord << endl;
    return 0;
}

// result:
Big Java 7th Ed | Horstman | 978-1118431115 | 99.85

Uncustomarily I am using CentOS currently, so my compiler version is below . The C++ version (g++), C++98 default:

g++ (GCC) 4.8.5 20150623 (Red Hat 4.8.5-4)
Copyright (C) 2015 Free Software Foundation, Inc.
This is free software; see the source for copying conditions.  There is NO
warranty; not even for MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.

Upvotes: 0

Volomike
Volomike

Reputation: 24916

If you're looking to replace more than a single character, and are dealing only with std::string, then this snippet would work, replacing sNeedle in sHaystack with sReplace, and sNeedle and sReplace do not need to be the same size. This routine uses the while loop to replace all occurrences, rather than just the first one found from left to right.

while(sHaystack.find(sNeedle) != std::string::npos) {
  sHaystack.replace(sHaystack.find(sNeedle),sNeedle.size(),sReplace);
}

Upvotes: 7

Lloydie
Lloydie

Reputation: 31

#include <iostream>
#include <string>
using namespace std;
// Replace function..
string replace(string word, string target, string replacement){
    int len, loop=0;
    string nword="", let;
    len=word.length();
    len--;
    while(loop<=len){
        let=word.substr(loop, 1);
        if(let==target){
            nword=nword+replacement;
        }else{
            nword=nword+let;
        }
        loop++;
    }
    return nword;

}
//Main..
int main() {
  string word;
  cout<<"Enter Word: ";
  cin>>word;
  cout<<replace(word, "x", "y")<<endl;
  return 0;
}

Upvotes: 3

Kirill V. Lyadvinsky
Kirill V. Lyadvinsky

Reputation: 99695

std::string doesn't contain such function but you could use stand-alone replace function from algorithm header.

#include <algorithm>
#include <string>

void some_func() {
  std::string s = "example string";
  std::replace( s.begin(), s.end(), 'x', 'y'); // replace all 'x' to 'y'
}

Upvotes: 983

UncleZeiv
UncleZeiv

Reputation: 18488

I thought I'd toss in the boost solution as well:

#include <boost/algorithm/string/replace.hpp>

// in place
std::string in_place = "blah#blah";
boost::replace_all(in_place, "#", "@");

// copy
const std::string input = "blah#blah";
std::string output = boost::replace_all_copy(input, "#", "@");

Upvotes: 151

T.E.D.
T.E.D.

Reputation: 44814

A simple find and replace for a single character would go something like:

s.replace(s.find("x"), 1, "y")

To do this for the whole string, the easy thing to do would be to loop until your s.find starts returning npos. I suppose you could also catch range_error to exit the loop, but that's kinda ugly.

Upvotes: 25

Konrad
Konrad

Reputation: 41027

As Kirill suggested, either use the replace method or iterate along the string replacing each char independently.

Alternatively you can use the find method or find_first_of depending on what you need to do. None of these solutions will do the job in one go, but with a few extra lines of code you ought to make them work for you. :-)

Upvotes: 4

Related Questions