Разбор только чисел из istream в C ++ - PullRequest
6 голосов
/ 19 августа 2011

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

(8,7,15)
(0,0,1) (0,3,2) (0,6,3)
(1,0,4) (1,1,5)

Мне нужно написать функцию, которая анализирует эти входы по одному разу, поэтому мне нужно разделить входные данные по номерам, например: 8, затем 7, затем 15, затем 0, еще один 0 и т. Д. .

Единственный способ, о котором я до сих пор думал, это использовать istream.get (), который возвращает код ASCII следующего символа, который я могу преобразовать обратно в его символьный формат, приведя его к типу char. Тогда я бы проверил, был ли символ цифрой или нет (поэтому скобки игнорируются), но таким образом любые двузначные (или тройные) цифры читаются только по одной цифре за раз.

Каков наилучший способ добиться этого?

Кстати, я должен использовать istream. Это часть спецификации, которую я не могу изменять

Спасибо

Ответы [ 5 ]

7 голосов
/ 19 августа 2011

Это одно решение:

struct integer_only: std::ctype<char> 
{
    integer_only(): std::ctype<char>(get_table()) {}

    static std::ctype_base::mask const* get_table()
    {
        static std::vector<std::ctype_base::mask> 
            rc(std::ctype<char>::table_size,std::ctype_base::space);

        std::fill(&rc['0'], &rc['9'+1], std::ctype_base::digit);
        return &rc[0];
    }
};

int main() {
        std::cin.imbue(std::locale(std::locale(), new integer_only()));
        std::istream_iterator<int> begin(std::cin);
        std::istream_iterator<int> end;
        std::vector<int> vints(begin, end);
        std::copy(vints.begin(), vints.end(), std::ostream_iterator<int>(std::cout, "\n"));
        return 0;
}

Введите:

(8,7,15)
(0,0,1) (0,3,2) (0,6,3)
(1,0,4) (1,1,5)

Выход:

8 7 15 0 0 1 0 3 2 0 6 3 1 0 4 1 1 5 

Онлайн демо: http://ideone.com/Lwx9y

Выше необходимо заменить std::cin потоком файла после успешного открытия файла, например:

 std::ifstream file("file.txt");
 file.imbue(std::locale(std::locale(), new integer_only()));
 std::istream_iterator<int> begin(file);
 std::istream_iterator<int> end;
 std::vector<int> vints(begin, end); //container of integers!

Здесь vints - вектор, содержащий все целые числа. Вы хотели бы поработать с vints, чтобы сделать что-то полезное. Также вы можете использовать его там, где ожидается int*:

void f(int *integers, size_t count) {}

f(&vints[0], vints.size()); //call a function which expects `int*`.

Подобный прием можно применять при чтении только слов из файла. Вот пример:

3 голосов
/ 19 августа 2011

Вот код, который вы можете адаптировать в соответствии с вашими потребностями

for (;;)
{
  int ch = in.get();
  if (ch == EOF)
    break;
  if (isdigit(ch))
  {
    int val = ch - '0';
    for (;;)
    {
      ch = in.get();
      if (!isdigit(ch))
        break;
      val *= 10;
      val += ch - '0';
    }
    // do something with val
  }
}

Это непроверенный код.

2 голосов
/ 19 августа 2011

Другое решение:

#include <string>
#include <ostream>
#include <fstream>
#include <iostream>

struct triple
{
    long a;
    long b;
    long c;
};

std::ostream& operator << (std::ostream& os, const triple& value)
{
    return os << value.a << "/" << value.b << "/" << value.c;
}

int main()
{
    std::ifstream stream("Test.txt");
    if (!stream)
    {
        std::cout << "could not open the file" << std::endl;
    }

    std::string dummy;
    triple value;
    while (std::getline(stream, dummy, '(') >> value.a &&
           std::getline(stream, dummy, ',') >> value.b &&
           std::getline(stream, dummy, ',') >> value.c)
    {
        std::cout << value << std::endl;
    }
}
2 голосов
/ 19 августа 2011

попробуй прочитать число.если это не помогло, очистите состояние ошибки и попробуйте прочитать char (и игнорируйте его).повторяйте эти два шага, пока чтение символа не завершится неудачно, и в этом случае вы находитесь в EOF или истинном сбое.

это может быть оптимизировано путем распознавания ')' и последующего чтения до '('.

но я не думаю, что оно того стоит.

ура & hth.,

1 голос
/ 19 августа 2011
int getFirstPos(const string& str)

{
int pos=0,PosHit=0;
bool bfind=false;
if((PosHit=str.find(','))!=string::npos){
    if(!bfind)  pos=PosHit;
    pos=pos>PosHit?PosHit:pos;
    bfind=true;
}
if((PosHit=str.find('('))!=string::npos){
    if(!bfind)  pos=PosHit;
    pos=pos>PosHit?PosHit:pos;
    bfind=true;
}
if((PosHit=str.find(')'))!=string::npos){
    if(!bfind)  pos=PosHit;
    pos=pos>PosHit?PosHit:pos;
    bfind=true;
}
return bfind?pos:string::npos;

}

void main()

{
    ifstream ifile("C:\\iStream.txt");
    string strLine;
    vector<double> vecValue;    //store the datas
    while(getline(ifile,strLine)){
        if(strLine.size()==0)
            continue;
        int iPos=0;
        while((iPos=getFirstPos(strLine))!=string::npos)
            strLine[iPos]=' ';
        istringstream iStream(strLine);
        double dValue=0;
        while(iStream>>dValue)
            vecValue.push_back(dValue);
    }
    //output the result!
    vector<double>::iterator it;
    for (it=vecValue.begin(); it!=vecValue.end()  ; ++it){
        cout<<setprecision(3)<<*it<<endl;
    }
}
...