在example(regex.cpp)中,库的作者为此结构创建了一个自定义结构(magic_number)和一个验证函数,以显示如何将自定义结构集成到程序选项中。我按照他的例子为自定义类(MyClass)创建了一个验证函数。编译器抱怨lecical_cast不适用于MyClass。然后我实现了std::istream& operator>>(std::istream& in, MyClass& d)
,删除了void validate(.., MyClass*, ..)
,代码编译。任何人都可以解释为什么该示例不需要operator>>
,而我的不需要validate
?
编辑:
#include <MyLib/MyClass.h>
std::istream& operator>>(std::istream& in, MyClass& obj) {
// some code to populate obj
return in;
}
po::variables_map parseCommandLine(int argc, char* argv[]) {
po::options_description options("Options");
options.add_options()
("help", "produce help message")
("obj", po::value<MyClass>(), "")
;
po::variables_map vm;
store(po::command_line_parser(argc, argv)
.options(options).run(), vm);
notify(vm);
return vm;
}
int main(int argc, char* argv[]) {
try {
po::variables_map vm = parseCommandLine(argc, argv);
MyClass obj = vm["my"].as<MyClass>();
cout << obj << endl;
} catch(std::exception& e) {
cout << e.what() << "\n";
return 1;
}
return 0;
}
我也试过对regex.cpp进行最小的更改:
#include <MyLib/MyClass.h>
编辑:添加validate
。他们都没有解决编译错误。
void validate(boost::any& v,
const std::vector<std::string>& values,
std::vector<MyClass>*, int)
{
}
void validate(boost::any& v,
const std::vector<std::string>& values,
MyClass*, long)
{
}
void validate(boost::any& v,
const std::vector<std::string>& values,
MyClass*, int)
{
}
编辑:它可能与命名空间有关。
在我通过namespace boost { namespace program_options { }}
包围验证函数后,编译的代码没有重载op&gt;&gt;。如果将validate放入与MyClass相同的命名空间中,它也可以工作。谁能解释一下呢?
答案 0 :(得分:5)
您面临的基本问题是C ++没有提供将字符串转换为任意用户对象的任何工具(我的意思是不编写任何代码)。
为了解决这个问题,program_options提供了两种可能性:
operator>>
,这是标准的C ++方式,但可能会对其他一些方面产生影响(即您可能希望以特定方式解析对象,但命令行除外)。在内部,boost::lexical_cast
用于实现转换,如果找不到op>>
则会抛出错误。validate
函数,该函数特定于program_options但在选项管理之外没有任何影响。我猜它会使用模板元编程来确定您是否提供了validate
,或者它将默认为lexical_cast
。
我无法帮助您为什么validate
尝试失败,因为您没有为其提供代码。
以下是一个工作示例:
#include <boost/program_options.hpp>
#include <vector>
#include <string>
namespace po = boost::program_options;
namespace lib {
class MyClass
{
public:
int a;
};
void validate(boost::any& v,
const std::vector<std::string>& values,
MyClass*, int)
{
po::validators::check_first_occurrence(v);
const string& s = po::validators::get_single_string(values);
v = boost::any(MyClass { boost::lexical_cast<int>(s) } );
}
}
po::variables_map parseCommandLine(int argc, char* argv[])
{
po::options_description options("Options");
options.add_options()
("help", "produce help message")
("obj", po::value<lib::MyClass>(), "")
;
po::variables_map vm;
store(po::command_line_parser(argc, argv)
.options(options).run(), vm);
notify(vm);
return vm;
}
int main(int argc, char* argv[])
{
try {
po::variables_map vm = parseCommandLine(argc, argv);
lib::MyClass obj = vm["obj"].as<lib::MyClass>();
cout << obj.a << endl;
} catch(std::exception& e) {
cout << e.what() << "\n";
return 1;
}
return 0;
}
使用命名空间,类和validate都必须属于同一个命名空间。