Eine kleine Ergänzung zum Artikel über die Verwendung von boost::program_options .
In diesem Anhang werde ich beschreiben, wie Aufzählungen verwendet werden. Weil sie nicht sofort funktionieren. Standardmäßig funktioniert boost::program_options nur mit Basistypen und einigen Typen aus der Standardbibliothek, wie z. B. std::string.
Nehmen wir an, wir haben eine Klasse, die die Aufzählung enum enthält.
class Logger { public: explicit Logger(); enum VerboseLevel { E_DEBUG = 0 E_INFO, E_WARNING, E_CRITICAL, }; };
Es ist logisch, dass wir diese Aufzählung im Befehlszeilenmenü verwenden möchten, um beispielsweise den Ausführlichkeitsgrad der Ausgabe an die Konsole oder an eine Datei festzulegen.
Das heißt, wir wollen unsere boost::program_options wie folgt konfigurieren.
m_desc.add_options() (MENU_ITEM_HELP.c_str(), "produce help message") ("level", po::value<Logger::VerboseLevel>(&m_verboseLevel)->default_value(Logger::E_WARNING), "set verbose level of logging, default level is Info\nLevels: \n0) Debug \n1) Info \n2) Warning \n3) Critical") ; po::store(po::parse_command_line(argc, argv, m_desc), m_vm); po::notify(m_vm);
Aber beim Kompilieren bekommen wir einen Fehler
Error C2338 Target type is neither std::istream`able nor std::wistream`able
In diesem Fall müssen wir den Operator operator>> überladen, um die benötigten Werte setzen zu können.
Die einfachste Variante könnte so aussehen
std::istream& operator>>(std::istream& in, Logger::VerboseLevel& verboseLevel) { std::string token; in >> token; if (token == "0") verboseLevel = Logger::VerboseLevel::E_DEBUG; else if (token == "1") verboseLevel = Logger::VerboseLevel::E_INFO; else if (token == "2") verboseLevel = Logger::VerboseLevel::E_WARNING; else if (token == "3") verboseLevel = Logger::VerboseLevel::E_CRITICAL; else in.setstate(std::ios_base::failbit); return in; }