Software-Entwickler und Programmierer möchten häufig Objekte in Strings konvertieren, zum Beispiel um ein Mitteilungsfeld mit der Anzahl der Wörter in einem Dokument anzuzeigen. Für diese Aufgabe gibt es mehrere Lösungsmöglichkeiten, wie Listing A zeigt: sprintf, ecvt oder die nicht-portable Funktion _itoa (aus Win32).
All diese Verfahren entstammen ursprünglich der Programmiersprache C, wobei jedes einzelne auch seine Nachteile hat:
- sprintf ist nicht typsicher, benötigt einen Puffer und ist schwierig in der Anwendung.
- Die Syntax von ecvt ist recht eigentümlich, schwer verständlich und funktioniert nur mit Zahlen.
- _itoa funktioniert nur mit Ganzzahlen und ist außerdem noch nicht einmal portabel.
In diesem Artikel wird eine einfache und effiziente Methode gezeigt, wie man jeden Datentyp in einen String verwandeln kann. Sie funktioniert für jeden Typ, der über einen überladenen Operator << verfügt. Und was noch besser ist: Es handelt sich um eine allgemeine Lösung, die mit jedem String-Typ funktioniert (std::string, std::wstring - Wide String – etc.).
Version 1.0: to_string
Um typsicher zu sein, muss die Methode mit allen Datentypen funktionieren. Daher empfiehlt sich der Einsatz von Templates:
Diese Lösung entstammt den STL-Streams (Standard Template Library). Außer File-Streams (basic_[i/o]fstream-Klassen) bietet STL auch String-Streams. Jedem String-Stream liegt eine Zeichenfolge zugrunde. Sobald man die Daten geschrieben hat, kann man sie als String behandeln und auf die Member-Funktion str() zugreifen, die einen std::(basic_)string ausgibt. Listing B zeigt die erste Version von to_string.
Die Funktion to_string funktioniert mit jedem Datentyp, der über einen überladenen Operator << verfügt. Dazu gehören die eingebauten Typen wie I und char.
Warum sollte man to_string benutzen?
Das Verhalten von to_string kann emuliert werden, wie Listing C zeigt. Es ist nur eine Geschmacksfrage, ob man to_string bevorzugt. Einer der Vorteile von to_string ist das Debugging. Wenn man eine std::string-Variable betrachtet, wird man auf jeden Fall einen const char *-Zeiger finden (einen String, wie zu erwarten). Dies ist für einen stringstream nicht immer der Fall, da die zugrunde liegende Implementierung mehrere Puffer enthalten kann, die nur zusammengefügt werden, wenn die Funktion str() aufgerufen wird.
Version 1.5: str_stream
Das Template to_string war schon ein Schritt in die richtige Richtung, aber eventuell möchte man ähnlich wie in Listing D gezeigt vorgehen. Allerdings ist diese Methode nicht portabel und möglicherweise gefährlich.
Etwas Ähnliches wie Listing D erreicht man mit einer Funktion, die einen Proxy mit einem zugrunde liegenden stringstream ausgibt. Alles, was an den Proxy geschrieben wird, wird auch an den stringstream geschrieben, wie Listing E zeigt. Dann werden die Daten implizit in einen String konvertiert (operator std::string()).
Falls man eine explizite Umwandlung wünscht, kann man as_string verwenden, inzwischen ohne Nebeneffekte (Listing F).
Die Implementierung von as_string kompliziert den Code allerdings wegen eines Bugs in Visual C++ 6. Der Code führt Folgendes aus:
- Bei einem Wert str_stream() << some_value wird const str_stream & ausgegeben (wie bei Streams üblich).
- Bei as_string (str_stream << … << as_string()) wird ein String ausgeben.
Da man weniger Code schreiben muss, wird man wahrscheinlich str_stream() gegenüber to_string() den Vorzug geben.
Neueste Kommentare
2 Kommentare zu Die perfekte Convert-to-string-Funktion in C++
Kommentar hinzufügenVielen Dank für Ihren Kommentar.
Ihr Kommentar wurde gespeichert und wartet auf Moderation.
Listing G
Beim Lesen fehlte mir "Listing G". Anscheinend ist kein Link vorhanden.
AW: Listing G
Vielen Dank für Ihren Hinweis. Das Listing ist jetzt korrekt verlinkt.