Небольшое дополнение к статье по использованию 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; }