Невеликий додаток до статті boost::program_options .
У цьому додатку опишу, як використовувати переліки enum. Бо одразу вони не працюють. За умовчанням boost::program_options працюють тільки з базовими типами та деякими типами зі стандартної бібліотеки, наприклад std::string.
Припустимо, ми маємо клас, який містить у собі перелік enum .
class Logger { public: explicit Logger(); enum VerboseLevel { E_DEBUG = 0 E_INFO, E_WARNING, E_CRITICAL, }; };
Логічно, що ми хотіли б використовувати цей перелік у меню командного рядка для встановлення рівня деталізації виведення в консоль або файл наприклад.
Тобто ми хочемо налаштувати наш boost::program_options .
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);
Але при складанні отримаємо помилку
Error C2338 Target type is neither std::istream`able nor std::wistream`able
В даному випадку нам необхідно зробити перевантаження оператора operator>> , щоб отримати можливість встановлювати необхідні значення.
Найпростіший варіант може виглядати так
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; }