8

该函数应该读取分数并将其放入数组中。如果用户输入“0”,则该函数应该退出。我正在尝试使用 cin.peek() 函数执行此操作,但执行始终进入 if 语句并且不允许用户退出。

我应该如何正确编码(我愿意不使用 peek(),我认为这是最简单的方法。)

谢谢!

void enterFrac(Fraction* fracs[], int& index)
    {
        int n, d;
        char c, slash;
        cout << "Enter fractions (end by entering a 0): ";
        c = cin.peek();

        if ( c != '0')
        {
            cin >> n >> slash >> d;
            Fraction* f = new Fraction();
            f->num = n;
            f->den = d;
            fracs[index] = f;
            index++;
        }
    }

然而,这个 peek() 测试有效:

#include <iostream>
using namespace std;

int main () {
  char c;
  int n;
  char str[256];

  cout << "Enter a number or a word: ";
  c=cin.peek();

  if ( (c >= '0') && (c <= '9') )
  {
    cin >> n;
    cout << "You have entered number " << n << endl;
  }
  else
  {
    cin >> str;
    cout << " You have entered word " << str << endl;
  }

  return 0;
}
4

1 回答 1

16

您使用 有两个问题std::istream::peek()

  1. 此函数访问下一个字符并且不跳过前导空格。您可能希望在确定下一个字符是什么之前跳过前导空格,例如,使用操纵器std::ws: (std::cin >> std::ws).peek()
  2. 的结果std::istream::peek()不是char. 相反,它是一个std::char_traits<char>::int_type(这是一个花哨的拼写int)。结果可能是,例如,std::char_traits<char>::eof()如果 的值'0'恰好为负(我不知道它所在的任何平台;但是,例如,我名字中的有趣字符在签名'ü'的平台上是负值)你char也不会得到正确的结果。也就是说,您通常将 的结果std::istream::peek()与 的结果进行比较std::char_traits<char>::to_int_type(),即,您会使用如下内容:std::cin.peek() == std::char_traits<char>::to_int_type('0')

也就是说,您的程序不会检查它是否可以成功读取由斜线分隔的提名者和分母。您总是想验证读取是否成功,例如,使用类似

if ((std::cin >> nominator >> slash >> denominator) && slash == '/') {
    ...
}

只是为了娱乐,您可以创建一个操纵器来测试字符是否是斜线,实际上:

std::istream& slash(std::istream& in) {
    if ((in >> std::ws).peek() != std::char_traits<char>::to_int_type('/')) {
        in.setstate(std::ios_base::failbit);
    }
    return in;
}

这样,您将封装斜线测试。如果您需要在多个地方使用它,这非常方便。

于 2012-11-21T21:04:08.120 回答