我有一个函数(遗留),它读取文件的前几行以确定其类型,然后关闭并重新打开文件,以便它可以使用正确的解释器重新读取整个文件。要点是:
void readFile(const char *filename) {
ifstream is(filename);
Filetype ft = determineFileType(is);
is.close();
is.open(filename);
parseFile(is, ft);
}
我需要一个类似的函数,可以在已经打开的流上工作。我创建了一个新函数,它接受一个ostream &
而不是一个char *filename
——基本上是这样的:
void readFile(istream &is) {
std::ios::streampos pos = is.tellg();
Filetype ft = determineFileType(is);
is.seekg(pos);
parseFile(is, ft);
}
当istream
实际上是stringstream
或fstream
时,它似乎起作用,但我想知道我是否只是幸运。我也做了一个小测试seekg
-ing std::cin
,它的工作,这让我很惊讶。
所以我的问题:什么类型的流是允许使用seekg
?它什么时候会失败?规范参考将是伟大的-我看了seekg
, pubseekpos
, seekpos
, seekoff
的东西都没有帮助。
我想用新的重新实现原来的函数(如下所示),但我只是不知道这是否安全。
void readFile(const char *filename) {
ifstream is(filename);
readFile(is);
is.close();
}
一个人能给出的唯一真正的答案是,它在它工作的地方工作。在在std::stringbuf
的情况下,它应该在任何地方工作。在…的情况下std::filebuf
,是否工作取决于系统;它将如果在实际文件上打开filebuf
,则通常工作,但将通常失败(如果系统不报告错误,则可能是静默的)对于许多其他类型的输入:从键盘,或命名管道,For的例子。
一个更健壮的解决方案是缓存初始输入,并从缓存中重新读取它。