为什么我的循环只从我的 .obj 文件中读取一个字符

Why does my loop only read one char from my .obj file?

本文关键字:我的 一个 字符 读取 循环 obj 文件 为什么      更新时间:2023-10-16

我正在尝试为Wavefront编写一个阅读器。OBJ 文件,以便我可以在 OpenGL 中绘制网格。为此,我尝试积累一些基本的C++文件 IO 知识,因为我主要了解 java。我从这个问题中了解到,>>可用于从流中读取数据。

我读取 OBJ 文件的策略是首先读取识别下一段数据的字符,然后继续尝试读取数字,直到我不能再读取。然后我尝试在下一个字符中阅读。我的代码如下:

std::ifstream objReader;
objReader.open("resources//file.obj");
char m;
int vindex = 0;
int iindex = 0; 
while (objReader >> m) {
    std::cout << "here" << std::endl;
    if (m == 'v') {
        GLfloat cv;
        while (objReader >> cv) {
            objData[vindex] = cv; //obj data is my verticies array to be drawn
            std::cout << objData[vindex] << std::endl;
            vindex++;
        }
    }
    else if (m == 'f') {
        GLuint ci;
        while (objReader >> ci) {
            indicies[iindex] = ci; //indicies is my EBO array 
            std::cout << indicies[iindex] << std::endl;
            iindex++;
        }
    } 
}

我正在使用的文件数据可以在这里找到。

现在,当我运行此代码时,它会很好地打开文件,并且可以在第一行中轻松读取。它将标记作为字符v然后将以下 3 个浮点存储到我的数组中。问题是,它就到此为止了。循环中断,甚至永远不会继续到第二行。不知何故,它无法再在文件中找到任何其他字符。为什么会发生这种情况,我该如何解决?

谢谢!

您的内部循环像这样解析浮点数:

while (objReader >> cv)

我假设这个GLfloatfloatdoubletypedef,根据您的描述。

此循环结束的唯一方法是当operator>>失败时。当operator>>失败时,它会将流置于错误状态。设置错误状态后,流上的所有后续操作都会自动失败。

因此,在此之后,当执行第二次返回到顶级循环时:

while (objReader >> m) {
现在,

此操作将立即失败,因为流现在处于错误状态。

您可以手动清除错误状态,但这并不是处理输入的干净方法。

如果始终需要三个浮点值,请将内部循环替换为迭代三次的for循环。

如果浮点变量的数量不同,我想你可以坚持你目前的方法,并通过在 while 循环后调用 clear() 方法来显式清除错误条件:

objReader.clear();

对于一个简单的解析要求,这可能已经足够好了......

没有必要重复Sam Varshavchik的回答,所以这里有一个解决方案:

  1. 将文件中的每一行读入 std::string 中,其中包含 std::getline
  2. 将行写入 std::stringstream
  3. 像现在一样从字符串流中读出字符并输入整数或浮点数大小写
  4. 从字符串流中读取浮点数的整数,直到行尾,就像你现在所做的那样。

应该有效的快速,未经测试的黑客示例:

std::ifstream objReader;
objReader.open("resources//file.obj");
char m;
int vindex = 0;
int iindex = 0; 
std::string line;
while (std::getline(objReader, line)) {
    std::cout << "here: " << line << std::endl;
    std::stringstream linereader(line);
    if (linereader >> m) // only enter if we read a character on the line
                         // testing here because we use the while to get a line 
    {// and continue as you did, but reading linereader instead of objReader
        if (m == 'v') { // 
            GLfloat cv;
            while (linereader >> cv) {
                objData[vindex] = cv; //obj data is my verticies array to be drawn
                std::cout << objData[vindex] << std::endl;
                vindex++;
            }
        }
        else if (m == 'f') {
            GLuint ci;
            while (linereader >> ci) {
                indicies[iindex] = ci; //indicies is my EBO array 
                std::cout << indicies[iindex] << std::endl;
                iindex++;
            }
        }
    }
}
相关文章: