пользовательские данные iostream - PullRequest
3 голосов
/ 01 сентября 2009

У меня есть структура данных, определенная как

struct myDataStruct
{
   int32_t header;
   int16_t data[8];
}

и я хочу взять символьный поток и превратить его в поток myData. Какой класс потока мне следует расширить? Я хотел бы создать собственный класс потока, чтобы я мог делать такие вещи как

myDataStruct myData;
myDataStruct myDataArray[10];

myDataStream(ifstream("mydatafile.dat"));
myDataStream.get(myData);
myDataStream.read(myDataArray, 10);

Ответы [ 2 ]

5 голосов
/ 01 сентября 2009

Вместо myDataStream.get(myData) перегрузка operator>> для вашего типа данных:

std::istream& operator>>(std::istream& is, myDataStruct& obj)
{
  // read from is into obj
  return is;
}

Если вы хотите читать в массив, просто напишите цикл:

for( std::size_t idx=0; idx<10; ++idx ) 
{
   myDataStruct tmp;
   if( is >> tmp )
     myDataArray[idx] = tmp;
   else
     throw "input stream broken!";
}

Используя шаблон функции, вы также можете перегрузить оператор для массивов с правой стороны (но я никогда не пробовал):

template< std::size_t N >
std::istream& operator>>(std::istream& is, myDataStruct (&myDataArray)[N])
{
  // use loop as above, using N instead of the 10
}

Но я не могу решить, великолепно это или подло.

0 голосов
/ 01 сентября 2009

Если вы работаете с неформатированным вводом, вам, вероятно, следует читать напрямую в двоичном виде. Обычно вы используете какую-то директиву, специфичную для компилятора, для создания структур данных без заполнения, а затем просто для чтения / записи из файла.

// Gcc
#pragma pack(1) // option 1: pragmas
struct frame {
   std::uint32_t header;
   std::uint16_t data[8];
} __attribute((packed)); // option 2: packed attribute
#pragma pack(0)

bool operator==( data const & lhs, data const & rhs )
{
   bool result = lhs.header == rhs.header;
   for ( int i = 0; i < 8; ++i )
   {
      result &= lhs.data[i] == rhs.data[i];
   }
   return result;
}

int main()
{
   frame data = { 10, 1, 2, 3, 4, 5, 6, 7, 8 };

   std::ofstream out( "data.bin", ofstream::binary );
   out.write( reinterpret_cast<char*>(&data), sizeof(data) );
   out.close();

   std::ifstream in( "data.bin", ifstream::binary );
   frame readed;
   in.read( reinterpret_cast<char*>(&readed), sizeof(readed) );
   in.close();

   std::cout << (readed == data) << std::endl; // 1
}

Директива компилятора для отключения заполнения для VS может отличаться (я считаю, что директива pragma работает как в gcc, так и в VS, но я всегда использовал этот атрибут).

...