basic_istream::seekg() 似乎不起作用

basic_istream::seekg() seems to not be working

以下程序应输出如下内容:

Begin found
space found
End found

但事实并非如此。

#include <sstream>
#include <istream>
#include <string>
#include <cctype>
#include <iostream>

bool Match(std::istream& stream, const std::string& str)
{
    std::istream::pos_type cursorPos = stream.tellg();

    std::string readStr(str.size(),'[=12=]');

    stream.read(&readStr[0],str.size());
    stream.seekg(cursorPos);
    if(std::size_t(stream.gcount()) < str.size() || readStr != str)
        return false;

    return true;
}

bool Take(std::istream& stream, const std::string& str)
{
    if(!Match(stream,str))
        return false;

    for(std::string::size_type i = 0; i < str.size(); ++i)
        stream.get();

    return true;
}

int main()
{
    std::string testFile = "BEGIN END";

    std::stringstream ss(testFile);
    auto c = ss.peek();
    while(!ss.eof() && ss.tellg() != -1)
    {
        if(Take(ss,"BEGIN"))
            std::cout << "Begin found" << std::endl;
        else if(Take(ss,"END"))
            std::cout << "End found" << std::endl;
        else if(std::isspace(c))
        {
            ss.get();
            std::cout << "space found" << std::endl;
        }
        else
            std::cout << "Something else found" << std::endl;
    }

    return 0;
}

我注意到它输出

Begin found
Something else found

当我单步执行调试器时,似乎当我到达 space 字符时,它首先检查是否有 Match()"BEGIN",它检索通过 tellg() 的光标位置,其值为 5。但是当它预期失败然后检查 Match()"END" 时,光标位于 -1,即结束。

看来 seekg() 调用没有工作,或者我没有正确使用它。

当您的程序进入主循环时,它首先以输入流和 "BEGIN" 作为参数执行 take。匹配 returns true,get 被调用 5 次,因为那是 BEGIN 的长度。

然后再次循环。它再次调用匹配。此时pos为5,BEGIN的长度。它尝试读取 len(BEGIN) 个字符,但您的字符串流没有那么多字符,因此它在位置 -1 处离开循环,并设置一个错误标志。

由于流处于错误状态,因此以下 seekg 调用没有达到预期的效果,无法解释程序的行为。