Boost parse_config_file, empty key value
Boost parse_config_file, empty key value
我使用Boost program_options以标准方式解析配置文件
显示在program_options的multiple_sources.cpp示例文件中。
ifstream ini_file("config.ini");
po::store(po::parse_config_file(ini_file, desc, true), vm);
po::notify(vm);
config.ini文件有空的键=值对,例如:
key1=value1
key2=value2
key3=
key4=
key5=value5
尝试读取此文件导致Boost错误:
boost::program_options::invalid_option_value
是否有任何机制在boost::program_options读取这样的配置文件与空的整体?
任何帮助都将非常感激。
我正在编辑这个问题,因为这个问题还没有解决。我只引用Boost(1.53)中的例子。
这是完整的multiple_sources.cpp文件:#include <boost/program_options.hpp>
namespace po = boost::program_options;
#include <iostream>
#include <fstream>
#include <iterator>
using namespace std;
// A helper function to simplify the main part.
template<class T>
ostream& operator<<(ostream& os, const vector<T>& v)
{
copy(v.begin(), v.end(), ostream_iterator<T>(os, " "));
return os;
}
int main(int ac, char* av[])
{
try {
int opt;
string config_file;
// Declare a group of options that will be
// allowed only on command line
po::options_description generic("Generic options");
generic.add_options()
("version,v", "print version string")
("help", "produce help message")
//("optimization","optimization level")
("config,c", po::value<string>(&config_file)->default_value("multiple_sources.cfg"),
"name of a file of a configuration.")
;
// Declare a group of options that will be
// allowed both on command line and in
// config file
po::options_description config("Configuration");
config.add_options()
("optimization", po::value<int>(&opt)->default_value(10),
"optimization level")
("include-path,I", po::value< vector<string> >()->composing(),
"include path")
;
// Hidden options, will be allowed both on command line and
// in config file, but will not be shown to the user.
po::options_description hidden("Hidden options");
hidden.add_options()
("input-file", po::value< vector<string> >(), "input file")
;
po::options_description cmdline_options;
cmdline_options.add(generic).add(config).add(hidden);
po::options_description config_file_options;
config_file_options.add(config).add(hidden);
po::options_description visible("Allowed options");
visible.add(generic).add(config);
po::positional_options_description p;
p.add("input-file", -1);
po::variables_map vm;
store(po::command_line_parser(ac, av).
options(cmdline_options).positional(p).run(), vm);
notify(vm);
ifstream ifs(config_file.c_str());
if (!ifs)
{
cout << "can not open config file: " << config_file << "n";
return 0;
}
else
{
store(parse_config_file(ifs, config_file_options), vm);
notify(vm);
}
if (vm.count("help")) {
cout << visible << "n";
return 0;
}
if (vm.count("version")) {
cout << "Multiple sources example, version 1.0n";
return 0;
}
if (vm.count("include-path"))
{
cout << "Include paths are: "
<< vm["include-path"].as< vector<string> >() << "n";
}
if (vm.count("input-file"))
{
cout << "Input files are: "
<< vm["input-file"].as< vector<string> >() << "n";
}
cout << "Optimization level is " << opt << "n";
}
catch(exception& e)
{
cout << e.what() << "n";
return 1;
}
return 0;
}
对应的配置文件(multiple_sources.cfg)为
#
# Comment out this line to use hard-coded default value of 10
#
optimization = 1
include-path = /opt
如果这个文件现在被修改为:
#
# Comment out this line to use hard-coded default value of 10
#
optimization =
include-path = /opt
抛出以下错误消息:
the argument for option 'optimization' is invalid
提出的验证重载等解决方案似乎不必要地复杂,特别是因为可能必须为每种数据类型编写验证函数,包含识别'n'其他空白的可能性。
有什么建议吗?谢谢大家抽出时间。
根据Aditya的建议,我替换了以下行:
("optimization", po::value<int>(&opt)->default_value(10),
"optimization level")
与以下内容:
("optimization", po::value<int>(&opt)->zero_tokens(),
"optimization level")
:
("optimization", po::value<int>(&opt)->implicit_value(10),
"optimization level")
,它们都不读取空白选项。Boost示例程序"parser_test.cpp"绕过zero_tokens()或implicit_value()的使用,将键值对读入vector,如下所示:
void test_config_file(const char* config_file)
{
options_description desc;
desc.add_options()
("gv1", new untyped_value)
("gv2", new untyped_value)
("empty_value", new untyped_value)
("plug*", new untyped_value)
("m1.v1", new untyped_value)
("m1.v2", new untyped_value)
("b", bool_switch())
;
const char content1[] =
" gv1 = 0#asdn"
"empty_value = n"
"plug3 = 7n"
"b = truen"
"[m1]n"
"v1 = 1n"
"n"
"v2 = 2n"
;
vector<option> a2 = parse_config_file<char>(config_file, desc).options;
BOOST_REQUIRE(a2.size() == 6);
check_value(a2[0], "gv1", "0");
check_value(a2[1], "empty_value", "");
check_value(a2[2], "plug3", "7");
check_value(a2[3], "b", "true");
check_value(a2[4], "m1.v1", "1");
check_value(a2[5], "m1.v2", "2");
}
我建议你使用try/catch这个异常,并只对那些你的程序真正需要的字段抛出错误,并且不能为空,否则该字段将被忽略。
不,目前没有办法在boost::program_options中处理这个问题。在INI文件中使用空键:值对与在命令行中指定选项而不提供参数相同。ypnos建议的编写自定义验证器的方法可能有效,但似乎不切实际,因为您需要将其应用于您期望的每个选项,而这些选项可能是空白的。您必须编写自己的po::parse_config_file实现,它将忽略没有值的行(除非相应的选项被标记为zero_token)以获得您正在寻找的结果。
我偶然发现了这个问题几次,最后通过在将其传递给program_options::parse_config_file
之前从初始ini文件中删除无效行来解决它。它可以通过在文件系统上创建一个临时修改的ini文件或将整个原始ini文件读取到内存并在那里修改它,然后创建istringstream
传递给parse_config_file
来完成。
我个人为此目的创建了一个boost::iostreams
过滤流类来动态地过滤ifstream
。这不会创建任何新的依赖,因为我已经使用了Boost,并且在我的情况下boost::iostreams
可以只使用头文件。
是这样使用的:
#include <fstream>
#include <boost/iostreams/filtering_stream.hpp>
#include <boost/program_options.hpp>
#include "inifilefilter.h" // IniFileInputFilter
...
std::ifstream file("config.ini");
if (file)
{
// filtering out empty values, i.e. name=
boost::iostreams::filtering_istream filteredIniFile;
filteredIniFile.push(IniFileInputFilter());
filteredIniFile.push(file);
po::parsed_options parsedFileOpts =
po::parse_config_file(filteredIniFile, configFileOptions, true);
...
}
我不会在这里显示inifilefilter.h
的代码,因为它有点冗长,所有的检查和角落案例,但同时它的想法很简单。IniFileInputFilter
的read()
方法缓冲每个输入行,然后在行尾进一步传递该行,如果它是有效的。'valid'表示它是[section]
行,或者在第一个'='
和行尾之间有一些非空格字符。
- 如果 KEY 是 std::list 或 std::vector 而不是值,那么 std::map 的默认行为是什么?
- 使用 Key 对 C++ 中的哈希映射进行排序. 无法排序
- 为什么"(!v.empty())"比"(v.size() >0)"好?
- std::<key-value>不同类型的对向量
- .value( "key" , default) 不适用于空的 json 对象吗?
- std::set<Key,Compare,Allocator>::find() 函数使用"<"运算符而不是"=="运算符背后的直觉是什么?
- std::move(key) 同时迭代unordered_map<字符串,字符串>?
- DirectX11 IASetVertexBuffers with nullptrs or empty buffers
- cpp / c ++中的grpc客户端代码,元数据x-api-key/x-goog-api-key不起作用,给了我语音A
- 为什么C++标准库容器函数 empty() 标记为 [[nodiscard]]?
- 黄金描述的C++ "Key Function"是什么?
- list::empty() 多线程行为?
- Qt3D:"Mesh is empty, nothing to load"消息
- 从更新查询获取'Cannot insert duplicate key'
- 使用迭代器成员函数是否仅适用于某些向量类型"empty()"?
- "Empty base optimization" lambda 捕获 - 标准禁止?为什么?
- QTcpSocket::readAll() is empty
- 在从 std::vector 移动的上调用 .clear()、.shrink_to_fit()、.empty() 是否合
- 如何计算C++中非重复值的数量 std::map<Key,Values>
- Boost parse_config_file, empty key value