输入流迭代器和异常

时间:2010-03-02 14:02:06

标签: c++ exception iterator istream

我几天前正在玩istream迭代器和异常处理,我偶然发现了这个好奇心:

#include <iostream>
#include <fstream>
#include <iterator>
#include <algorithm>

using namespace std;

int main(int argc, char* argv[])
{
   if (argc < 2) {
      cout << argv[0] << " <file>" << endl;
      return -1;
   }

   try {
      ifstream ifs(argv[1]);
      ifs.exceptions(ios::failbit | ios::badbit);
      istream_iterator<string> iss(ifs), iss_end;
      copy(iss, iss_end, ostream_iterator<string>(cout, "\n"));
   }
   catch (const ios_base::failure& e) {
      cerr << e.what() << endl;
      return -2;
   }

   return 0;
}

为什么在读取输入文件的最后一个单词后总会引发failbit异常?

3 个答案:

答案 0 :(得分:3)

只要读取操作无法提取任何字符,就会设置

failbit,无论这是因为它是否触及EOF。

stringstream ss ("foo");
string s;
int i;

ss >> i; // sets failbit because there is no number in the stream
ss.clear();
ss >> s; // sets eofbit because EOF is hit
ss.clear();
ss >> s; // sets eofbit and failbit because EOF is hit and nothing is extracted.

答案 1 :(得分:0)

通过读取直到失败 - 触发异常 - 然后检查失败原因来检测EOF条件。

要展开:当使用&gt;&gt;读取值后,istream_iterator变为无效,流运算符void *返回NULL。但为此,运营商&gt;&gt;必须设置失败位,因此提高异常。

答案 2 :(得分:0)

好问题。能够在该呼叫中捕获其他故障会很好,但是当它达到eof时它会继续正常运行。

那就是说,我之前没有使用过流的异常。我认为你可以复制并检查流的状态,然后检测其他错误,例如:

ifstream ifs(argv[1]);
if (!ifs) {
    cerr << "Couldn't open " << argv[1] << '\n';
    return -1;
}
//ifs.exceptions(ios::failbit | ios::badbit);
istream_iterator<std::string> iss(ifs), iss_end;
copy(iss, iss_end, ostream_iterator<std::string>(cout, "\n"));
if (!ifs.eof()) {
    cerr << "Failed to read the entire file.\n";
    return -2;
}