10

假设我有如下输入:

N (X_1,Y_1) (X_2,Y_2) .... (X_N, Y_N)

其中 N、X_i 和 Y_i 是整数。

一个例子:

2 (55,1) (521,7)

要阅读这篇文章,我可以这样做(假设所有变量都已定义,等等):

fscanf(fin,"%d ",&N);
for (int i = 0; i < N; i++)
   fscanf(fin,"(%d,%d) ", &X[i], &Y[i]);

问题是,如何使用 ifstream 轻松做到这一点。我可以得到字符串,然后我可以摆脱非数字并使用字符串流我可以得到两个数字,但这似乎有点麻烦。有没有更简单、更优雅的方法?

谢谢

4

2 回答 2

7
int n, x, y;
char c;
if (is >> n)
    for (int i = 0; i < n; ++i)
        if (is >> c && c == '(' &&
            is >> x &&
            is >> c && c == ',' &&
            is >> y &&
            is >> c && c == ')')
        {
            X[i] = x;
            Y[i] = y;
        }
        else
            throw std::runtime_error("invalid inputs");

您可以将上面最重要的内部if条件简化为...

is >> chlit('(') >> x >> chlit(',') >> y >> chlit(')')

...具有用于使用特定字符的简单支持类型:

struct chlit
{
    chlit(char c) : c_(c) { }
    char c_;
};

inline std::istream& operator>>(std::istream& is, chlit x)
{
    char c;
    if (is >> c && c != x.c_)
        is.setstate(std::iostream::failbit);
    return is;
}

在 ideone 上查看一个完整的程序来说明这一点。

我的一篇旧文章为使用特定字符串做了类似的事情。(上面chlit可能是一个模板,但chlit<','>()很难读和写——我宁愿相信编译器)。

于 2011-01-17T08:38:38.733 回答
3
cin >> N;
for (int i = 0; i < N; i++)
{
    cin.ignore(100,'(');
    cin >> X[i];
    cin.ignore(100,',');
    cin >> Y[i];
    cin.ignore(100,')');
}

它也可以处理空格,因为它可以读取如下输入:

2  (  1  ,  3  )    (  5  ,  6  )

ideone 演示:http ://www.ideone.com/hO0xG

于 2011-01-17T09:29:39.790 回答