2

我正在尝试使用 Boost Spirit 库编写一个基本的 OBJ 文件加载器。虽然我使用标准 std::ifstreams 让它工作,但我想知道是否可以使用内存映射文件对整个文件执行短语解析,因为它似乎提供了此处发布的最佳性能。

我有以下代码,它似乎运行良好,但是当文件中有注释时它会中断。所以,我的问题是如何使用 Spririt 忽略 OBJ 文件中以“#”开头的注释?

struct vertex {
    double x, y, z;
};

BOOST_FUSION_ADAPT_STRUCT(
                          vertex,
                          (double, x)
                          (double, y)
                          (double, z)
                          )
std::vector<vertex> b_vertices         
boost::iostreams::mapped_file mmap(
                                           path,
                                           boost::iostreams::mapped_file::readonly);
        const char* f = mmap.const_data();
        const char* l = f + mmap.size();


        using namespace boost::spirit::qi;

      bool ok = phrase_parse(f,l,(("v" >> double_ >> double_ >> double_) |
                               ("vn" >> double_ >> double_>> double_)) % eol ,
                               blank, b_vertices);

当没有注释或除顶点/法线之外的任何其他数据时,上述代码运行良好。但是当有不同类型的数据时,解析器会失败(因为它应该),我想知道是否有一种方法可以让它工作而无需返回解析每一行,因为它更慢(在我的测试中几乎是 2.5 倍) . 谢谢!

4

2 回答 2

3

想到的最简单的方法是简单地使注释可跳过:

bool ok = qi::phrase_parse(
        f,l,
         (
               ("v"  >> qi::double_ >> qi::double_ >> qi::double_) |
               ("vn" >> qi::double_ >> qi::double_ >> qi::double_)
          ) 
          % qi::eol,
        ('#' >> *(qi::char_ - qi::eol) >> qi::eol | qi::blank), b_vertices);

请注意,如果#出现在行内的某处,这也会“识别”注释。这可能很好(因为它会使解析失败,除非它是在其他有效输入行上尾随的注释)。

在 Coliru 上看到它

或者,使用一些凤凰魔法来处理“注释行”,就像处理“vn”或“v”行一样。

于 2013-12-30T17:33:44.087 回答
0

我意识到我的评论/帖子不是直接相关的代码,但如果可能的话,我不打算重新发明轮子,我想知道这个库。我正在使用手写的 OBJ/Wavefront 加载器,但在我的研究中,我发现了这个库Tiny Obj Loader。这个库是用 C++ 编写的,除了 C++ STL 之外没有依赖项。它很好地处理了 Wavefront 规范的边缘情况,而且速度非常快。用户要做的就是将 Tiny OBJ 对象转换为他们的代码。TinyObjLoader 也被很多项目采用。对于没有直接回答这个问题,我深表歉意,我的愿望是了解这个伟大的图书馆。

于 2016-07-07T19:44:10.767 回答