具有增强精神的 OBJ 解析器 - 忽略评论

OBJ Parser with Boost Spirit - Ignoring comments

本文关键字:评论 OBJ 增强      更新时间:2023-10-16

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

我有以下代码,它似乎运行良好,但是当文件中有注释时它会中断。所以,我的问题是你如何使用 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 倍)。谢谢!

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

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);

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

在科里鲁现场观看

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

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