Проблема с ++ при чтении файла с помощью getline () и правильном форматировании вывода - PullRequest
0 голосов
/ 11 ноября 2018

Используя приведенный ниже код, я читаю файл .txt и не могу его правильно вывести. Он либо не правильно разбивает первый ввод (со всеми разделителями запятых), либо неправильно, если я использую разделитель запятых только после имени элемента (2-й элемент в текстовом файле).

Вот код:

#include <iostream>
#include <fstream>
using namespace std;

int main()
{
    ifstream inventoryFile;
    string itemType[100], itemName[100], matCost[100], manHours[100];
//  float matCost[100], manHours[100];
    string temp;

    inventoryFile.open("items2.txt");

    cout << endl << endl;

    if(!inventoryFile)
    {
        cout << "\n\nCan't read from file.\n";
        exit(1);
    }
    int numItems=0;

    getline(inventoryFile, temp, ',');
    while(!inventoryFile.eof())
    {
        itemType[numItems] = temp;
        inventoryFile >> itemName[numItems];
        inventoryFile.ignore();
        inventoryFile >> matCost[numItems];
        inventoryFile.ignore();
        inventoryFile >> manHours[numItems];
        inventoryFile.ignore();
        numItems++;
        getline(inventoryFile, temp, ',');
    }

    for(int j=0; j<numItems; j++)
    {
        cout << "Item Type:\t" << itemType[j] << endl;
        cout << "ITem Name:\t" << itemName[j] << endl;
        cout << "Mats Cost:\t" << matCost[j] << endl;
        cout << "Man Hours:\t$" << manHours[j] << endl << endl; 
    }

    return 0;
}

Файл items.txt:

1, Xiphos, 7.46, 2
2, Dao, 3.45, 2.7
3, Jian, 2.31, 0.5
1, Rapier, 8.32, 2.3
2, Hook Sword, 2.11, 0.75
1, Panzerstecher, 2.23, 1.25
2, Kopis, 14.89, 2.3
3, Longsword, 5.43, 0.5
1, Tuck, 2.5, 15
1, Small Sword, 7.5, 2
3, Broadsword, 0.5, 0.25

Файл items2.txt:

1 Xiphos, 7.46 2
2 Dao, 3.45 2.7
3 Jian, 2.31 0.5
1 Rapier, 8.32 2.3
2 Hook Sword, 2.11 0.75
1 Panzerstecher, 2.23 1.25
2 Kopis, 14.89 2.3
3 Longsword, 5.43 0.5
1 Tuck, 2.5 15
1 Small Sword, 7.5 2
3 Broadsword, 0.5 0.25

Ответы [ 2 ]

0 голосов
/ 11 ноября 2018

Для items.txt это должно работать:

#include <cstdlib>
#include <fstream>
#include <iostream>
#include <sstream>
#include <vector>

using namespace std;

struct Row {
    int itemType;
    string itemName;
    float matCost;
    float manHours;
};

istream& operator>>(istream& in, Row& row){
    char comma;
    char firstCharOfName;
    in >> row.itemType >> comma >> firstCharOfName;
    in.putback(firstCharOfName);
    getline(in, row.itemName, ',');
    return in >> row.matCost >> comma >> row.manHours;
}

int main() {
    vector<Row> table;
    {
        ifstream inventoryFile("item.txt");
        string line;
        cout << "\n\n";

        if (!inventoryFile) {
            cerr << "\n\nCan't read from file.\n";
            return EXIT_FAILURE;
        }
        while (getline(inventoryFile, line)) {
            // make room for next row
            istringstream iss(line);
            table.resize(table.size() + 1U);
            Row &newRow = table.back();
            iss >> newRow;
            if (!iss) {
                // skip row on error
                table.resize(table.size() - 1U);
            }
        }
    }

    for (int j = 0; j < table.size(); j++) {
        cout << "Item Type:\t" << table[j].itemType << '\n'
             << "ITem Name:\t" << table[j].itemName << '\n'
             << "Mats Cost:\t$" << table[j].matCost << '\n'
             << "Man Hours:\t" << table[j].manHours << "\n\n";
    }

    return EXIT_SUCCESS;
}

PS: перемещено $ в Mats Cost.

0 голосов
/ 11 ноября 2018

Это должно работать для формата в items2.txt.

string itemType[100], itemName[100];
float matCost[100], manHours[100];
//...
//getline(inventoryFile, temp, ',');
// get itemType and the whitespace after it
while(inventoryFile >> itemType[numItems] >> std::ws)
{
    std::getline(inventoryFile, itemName[numItems], ',');
    inventoryFile >> matCost[numItems] >> manHours[numItems];
    // don't count this entry if the stream is in a failed state
    if(inventoryFile.fail()) break;
    inventoryFile.ignore();
    ++numItems;
}
...