В приведенном ниже коде я использовал параметры программы для чтения параметров из командной строки или файла. Кроме того, параметры могут быть установлены программно во время выполнения через ConfigProxy :: setConfig
po::options_description desc("Allowed options");
desc.add_options()
...
("compression", po::value<int>(), "set compression level");
po::variables_map vm;
class ConfigProxy
{
template< typename T>
void setConfig( const std::string key, const T value ){
... // check if the key exists in variable map "vm"
// key exists, set the value
runtimeConfig[key] = po::variable_value( boost::any(value), false);
}
po::variable_value& operator[] (const std::string key) const{
...
// if exists in runtimeConfig return the value in runtimeConfig
// of type program_options::variable_value
...
// else return value in variable map "vm"
}
std::map<std::string, boost::program_options::variable_value> runtimeConfig;
}
через ConfigProxy, значение параметра получается
if( vm.count("compression") ){
int value = proxyConfig["compression"].as<int>();
...
}
Однако, если предоставленное пользователем значение параметра «сжатие» имеет значение , неправильный тип , например
configProxy.setConfig("compression", "12" );
...
int value = configProxy["compression"].as<int>(); // was set as string
тогда генерируется исключение
what(): boost::bad_any_cast: failed conversion using boost::any_cast
Исключение ясно показывает проблему приведения типа. Но сообщение кажется не очень полезным для пользователей, чтобы выяснить , какая опция отвечает за ошибку.
Есть ли лучший способ информировать пользователей об этом типе ошибки, вместо того, чтобы выдавать bad_any_cast исключение?
----- Редактировать --------------------------
Благодаря Люку Дантону и Тони я обнаружил, как в параметрах программы отображаются ошибки.
void validate(boost::any& v,
const std::vector< std::basic_string<charT> >& xs,
T*, long)
{
validators::check_first_occurrence(v);
std::basic_string<charT> s(validators::get_single_string(xs));
try {
v = any(lexical_cast<T>(s));
}
catch(const bad_lexical_cast&) {
boost::throw_exception(invalid_option_value(s));
}
}
Я думаю, что, реализовав логику, я смогу избавиться от исключения bad_any_cast.