Чтение точек из потока, состоящего только из одной строки - PullRequest
0 голосов
/ 15 января 2019

У меня есть файл, который сначала сообщает мне, сколько точек я буду читать в следующей строке. Так, например, мой файл выглядит так:

7
a,b c,d e,f g,h, i,j k,l m,n

Итак, я знаю, что следующая строка после 7 - это 7 пар целых чисел, разделенных запятой, и каждая пара, разделенная пробелом.

Что я хочу: иметь вектор из 7-ти точечных элементов.

У меня есть класс с именем Point:

class Point {
public:
    int x;
    int y;
    bool operator==(const Point q){
        return (q.x == this->x && q.y == this->y);
    }
};

Поэтому, когда я читаю этот файл, я хотел бы иметь вектор V, где:

V[0].x = a
V[0].y = b
V[1].x = c
V[1].y = d

и т. Д.

Я могу прочитать 7 штрафа, но как мне прочитать каждую из 7 пар целых чисел по отдельности? Мне это нужно, потому что я собираюсь хранить (a, b) (c, d) ... в векторе.

Это не только 2 балла. Первая строка файла сообщает мне, сколько очков я собираюсь сохранить.

Они не читаются со стандартного ввода.

Они читаются из файла.

Я пытался использовать sscanf, но я думаю, что это только для случаев, когда у вас есть несколько строк с этой информацией, и я бы не хотел изменять свой формат.

Это то, что я имею до сих пор:

void process_file(string filename){
    ifstream thracklefile;
    string line;
    int set_size;
    thracklefile.open(filename);

    getline(thracklefile,line); //store set size.
    set_size = stoi(line);

    //Store points in following line
    points.clear();
    points.resize(set_size);
    getline(thracklefile,line); //store the points.
    }

Я не хочу игнорировать запятые, каждая запятая является частью информации, которую я хочу хранить для каждой точки.

1 Ответ

0 голосов
/ 15 января 2019

Я думаю, что большая часть обсуждения в комментариях посвящена семантике. Рекомендуется "игнорировать" запятые, но вы не можете сделать это, поскольку они находятся в файле. Возможно, лучшим термином является «отбросить». Слово "игнорировать" используется, так как есть функция Cost iostream ignore.

Есть много способов справиться с этим. Один из вариантов - переопределить операторы вставки / извлечения потока:

class Point {
public:
    int x;
    int y;

    // Don't really need this as members are public, but
    // in case you change that in the future....
    friend istream& operator>>(istream& in, Point& p);
    friend ostream& operator<<(ostream& out, const Point& p);
};

istream& operator>>(istream& in, Point& p)
{
    char separator;
    // Try to read <int><char><int>
    in >> p.x >> separator >> p.y;
    // The stream may be in an error state here. That
    // is ok. Let the caller handle that
    // Also note that we discard (ignore) "separator"
    return in;
}
ostream& operator<<(ostream& out, const Point& p)
{
    out << p.x << ',' << p.y;
    return out;
}

int main() {
    int num_points;
    std::cin >> num_points;
    Point p;
    for (int i = 0; i < num_points; i++) {
        if (!(std::cin >> p)) {
            // There was an error
            std::cout << "File format error!" << std::endl;
            break;
        }
        std::cout << p << std::endl;
    }

    return 0;
}

В примере используется cin, но любой поток должен работать, включая ifstream.

...