I am trying to parse a c++ string on every ‘^’ character into a vector tokens. I have always used the boost::split method, but I am now writing performance critical code and would like to know which one gives better performance.
For example:
string message = "A^B^C^D";
vector<string> tokens;
boost::split(tokens, message, boost::is_any_of("^"));
vs.
boost::char_separator<char> sep("^");
boost::tokenizer<boost::char_separator<char> > tokens(text, sep);
Which one would give better performance and why?
The best choice depends on a few factors. If you’re only needing to scan the tokens once, then the boost::tokenizer is a good choice in both runtime and space performance (those vectors of tokens can take up a lot of space, depending on input data.)
If you’re going to be scanning the tokens often, or need a vector with efficient random access, then the boost::split into a vector may be the better option.
For example, in your “A^B^C^…^Z” input string where the tokens are 1-byte in length, the
boost::split/vector<string>method will consume at least 2*N-1 bytes. With the way strings are stored in most STL implementations you can figure it taking more than 8x that count. Storing these strings in a vector is costly in terms of memory and time.I ran a quick test on my machine and a similar pattern with 10 million tokens looked like this:
If you’re just doing a one-time scan of the tokens, then clearly the tokenizer is better.
But, if you’re shredding into a structure that you want to reuse during the lifetime of your application, then having a vector of tokens may be preferred.
If you want to go the vector route, then I’d recommend not using a
vector<string>, but a vector of string::iterators instead. Just shred into a pair of iterators and keep around your big string of tokens for reference. For example:This improved version takes 1.6s and 390MB on the same server and test. And, best of all the memory overhead of this vector is linear with the number of tokens — not dependent in any way on the length of tokens, whereas a
std::vector<string>stores each token.