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

basic_istream::seekg() seems to not be working

本文关键字:不起作用 seekg istream basic      更新时间:2023-10-16

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

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(),'');
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

当我逐步使用调试器时,似乎当我达到空格字符时,它首先检查是否有要"BEGIN"Match(),它通过tellg()检索光标位置,其值为5。但是当它意外失败,然后用"END"检查Match()时,光标在-1,即结束。

因此,seekg()调用似乎不起作用,或者我没有正确使用它。

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

然后它再次通过循环。 它再次调用匹配。 此时 pos 为 5,即 BEGIN 的长度。 它尝试读取 len(BEGIN( 字符,但字符串流没有那么多字符,因此它将循环保留在位置 -1,并设置错误标志。

由于流是错误状态,因此以下 seekg 调用不会产生预期的效果,从而解释程序的行为。