3.9 KiB
~english \page chunk_stream Chunk stream and versioned serialization ~russian \page chunk_stream Поток чанков и версионная сериализация
~english
\a PIChunkStream is a binary stream where data is stored as chunks: each chunk has an integer \e id and a value. Reading is id-based, so you can add or reorder fields over time and stay backward compatible: old readers ignore unknown ids, new readers can skip optional ids.
Two format versions exist (\a PIChunkStream::Version_1 and \a Version_2); the writer chooses the version, the reader detects it automatically. By default new data is written in Version_2.
When to use
Use chunk streams when:
- You need to extend structures without breaking existing stored data (add new fields with new ids).
- You want optional or reordered fields in a single stream.
- You use \ref code_model to generate serialization: with default (chunk) mode, \c pip_cmg emits operators that read/write via \a PIChunkStream and field ids (see \ref code_model "code_model" for PIMETA \c id and \c simple-stream / \c no-stream).
For fixed, non-extensible layouts, plain \a PIBinaryStream operators (see \ref iostream) are enough.
Usage
Build a \a PIChunkStream from a \a PIByteArray (read or read-write). Write with \c cs << cs.chunk(id, value) or \c add(id, value); read with \c read() to get the next id, then \c get(value). Call \c data() to get the byte buffer for storing or sending. Example (conceptually as in code_model output):
\code{.cpp} PIByteArray buf; PIChunkStream cs(&buf); cs << cs.chunk(1, i) << cs.chunk(2, s); // later: PIChunkStream reader(buf); while (!reader.atEnd()) { switch (reader.read()) { case 1: reader.get(i); break; case 2: reader.get(s); break; } } \endcode
Generated operators for structs use the same pattern; see \ref code_model.
~russian
\a PIChunkStream — бинарный поток, в котором данные хранятся чанками: у каждого чанка целочисленный \e id и значение. Чтение идёт по id, поэтому можно добавлять или менять порядок полей с сохранением обратной совместимости: старые читатели игнорируют неизвестные id, новые могут пропускать необязательные.
Есть две версии формата (\a PIChunkStream::Version_1 и \a Version_2); версию выбирает запись, при чтении она определяется автоматически. По умолчанию запись идёт в Version_2.
Когда использовать
Имеет смысл использовать поток чанков, когда:
- Нужно расширять структуры без поломки уже сохранённых данных (новые поля — новые id).
- Нужны необязательные или переставляемые поля в одном потоке.
- Используется \ref code_model для генерации сериализации: в режиме по умолчанию (chunk) \c pip_cmg выдаёт операторы через \a PIChunkStream и id полей (см. \ref code_model по PIMETA \c id и \c simple-stream / \c no-stream).
Для фиксированных неизменяемых форматов достаточно обычных операторов \a PIBinaryStream (см. \ref iostream).
Использование
Создают \a PIChunkStream из \a PIByteArray (чтение или чтение/запись). Запись: \c cs << cs.chunk(id, value) или \c add(id, value); чтение: \c read() — следующий id, затем \c get(value). \c data() возвращает буфер для сохранения или передачи. Примеры генерации операторов — в \ref code_model.