After four months of intensive development work, I am happy to announce that the first QStringTokenizer commits have landed in what will eventually become Qt 6.0. The docs should show up, soon.
While the version in Qt will be Qt 6-only, KDAB will release this tool for Qt 5 as part of its KDToolBox productivity suite. Yes, that means the code doesn't require C++17 and works perfectly fine in pure C++11.
This is a good time to recapitulate what QStringTokenizer is all about.
QStringTokenizer: The Zero-Allocation String Splitter
Three years ago, when QStringView was first merged for Qt 5.10, I already wrote that we wouldn't want to have a method like QString::split() on QStringView. QStringView is all about zero memory allocations, and split() returns an owning container of parts, say QVector, allocating memory.
So how do you return the result of string splitting, if not in a container? You take a cue from C++20's std::ranges and implement a Lazy Sequence. A Lazy Sequence is like a container, except that it's elements aren't stored in memory, but calculated on the fly. That, in C++20 coroutine terms, is called a Generator.
So, QStringTokenizer is a Generator of tokens, and, apart from its inputs, holds only constant memory.
Here's the example from 2017, now in executable form:
Except, we beefed it up some:
Oh, and this also works now:
QStringTokenizer: The Universal String Splitter
When I initially conceived QStringTokenizer in 2017, I thought it would just work on QStringView and that'd be it. But the last example clearly shows that it also supports splitting QLatin1String. How is that possible?
This is where C++17 comes in, on which Qt 6.0 will depend. C++17 brought us Class Template Argument Deduction (CTAD):
And that's what we used in the examples above. In reality, QStringTokenizer is a template, but the template arguments are deduced for you.
So, this is how QStringTokenizer splits QStrings as well as QLatin1Strings: in the first case, it's QStringTokenizer<QStringView, QChar>, in the second, QStringTokenizer<QLatin1String, QChar>. But be warned: you should never, ever, explicitly specify the template arguments yourself, as you will likely get it wrong, because they're subtle and non-intuitive. Just let the compiler do its job. Or, if you can't rely on C++17, yet, you can use the factory function qTokenize():
QStringTokenizer: The Safe String Splitter
One thing I definitely wanted to avoid is dangling references a la QStringBuilder:
But since the ranged for loop there is equivalent to
if QStringTokenizer simply operated on QStringView or QLatin1String, the following would happen: The __range variable keeps the QStringTokenizer object alive throughout the for loop (ok!), but the temporary returned from widget->text() would have been destroyed in line 3, even before we enter the for loop (oops).
This is not desirable, but what can we do against it? The solution is as simple as it is complex: detect temporaries and store them inside the tokenizer.
Yes, you heard that right: if you pass a temporary ("rvalue") owning container to QStringTokenizer, the object will contain a copy (moved from the argument if possible) to extend the string's lifetime to that of the QStringTokenizer itself.
Future
Now that we have developed the technique, we very strongly expect it to be used in Qt 6.0 for QStringBuilder, too.
By Qt 6.0, we expect QStringTokenizer to also handle the then-available QUtf8StringView as haystack and needle, as well as QRegularExpression and std::boyer_moore_searcher and std::boyer_moore_horspool_searcher as needles. We might also re-implement it as a C++20 coroutine on compilers that support them, depending on how much more performance we'll get out of it.
Conclusion
QStringTokenizer splits strings, with zero memory allocations, universally, and safely. Get it for free right now from KDToolBox, and you can future-proof your code with an eye towards Qt 6.
Trusted software excellence across embedded and desktop platforms
The KDAB Group is a globally recognized provider for software consulting, development and training, specializing in embedded devices and complex cross-platform desktop applications. In addition to being leading experts in Qt, C++ and 3D technologies for over two decades, KDAB provides deep expertise across the stack, including Linux, Rust and modern UI frameworks. With 100+ employees from 20 countries and offices in Sweden, Germany, USA, France and UK, we serve clients around the world.