git-svn-id: svn://db.shs.com.ru/libs@622 a8b55f48-bf90-11e4-a774-851b48703e85

This commit is contained in:
2019-10-28 09:10:38 +00:00
parent 58d504dfb2
commit 7d360917f5
2 changed files with 115 additions and 5 deletions

View File

@@ -23,6 +23,32 @@ void ChunkStream::setSource(QDataStream & str) {
} }
int ChunkStream::read() {
switch (version_) {
case Version_1:
stream_ >> last_id >> last_data;
break;
case Version_2:
last_id = readVInt(stream_);
last_data.resize(readVInt(stream_));
//piCout << last_id << last_data.size();
stream_.readRawData(last_data.data(), last_data.size());
break;
default: break;
}
return last_id;
}
void ChunkStream::readAll() {
data_map.clear();
while (!atEnd()) {
read();
data_map[last_id] = last_data;
}
}
ChunkStream::~ChunkStream() { ChunkStream::~ChunkStream() {
} }
@@ -35,4 +61,53 @@ void ChunkStream::_init() {
buffer.setBuffer(data_); buffer.setBuffer(data_);
buffer.open(QIODevice::ReadWrite); buffer.open(QIODevice::ReadWrite);
stream_.setDevice(&buffer); stream_.setDevice(&buffer);
if (!data_->isEmpty()) {
uchar v = data_->at(0);
if ((v & 0x80) == 0x80) {
v &= 0x7f;
switch (v) {
case 2: version_ = (uchar)Version_2; stream_.skipRawData(1); break;
default: break;
}
} else
version_ = Version_1;
}
}
uint ChunkStream::readVInt(QDataStream & s) {
if (s.atEnd()) return 0;
uchar bytes[4]; s >> bytes[0];
uchar abc = 0;
for (abc = 0; abc < 3; ++abc) {
uchar mask = (0x80 >> abc);
if ((bytes[0] & mask) == mask) {
//if (s.isEmpty()) return 0;
bytes[0] &= (mask - 1);
s >> bytes[abc + 1];
} else break;
}
uint ret = 0;
for (int i = 0; i <= abc; ++i) {
ret += (bytes[i] << (8 * ((int)abc - i)));
}
return ret;
}
void ChunkStream::writeVInt(QDataStream & s, uint val) {
if (val > 0xfffffff) return;
if (val <= 0x7f) {
s << uchar(val);
return;
}
if (val <= 0x3fff) {
s << uchar((val >> 8) | 0x80) << uchar(val & 0xff);
return;
}
if (val <= 0x1fffff) {
s << uchar((val >> 16) | 0xc0) << uchar((val >> 8) & 0xff) << uchar(val & 0xff);
return;
}
s << uchar((val >> 24) | 0xe0) << uchar((val >> 16) & 0xff) << uchar((val >> 8) & 0xff) << uchar(val & 0xff);
} }

View File

@@ -4,14 +4,20 @@
#include <QDataStream> #include <QDataStream>
#include <QBuffer> #include <QBuffer>
#include <QByteArray> #include <QByteArray>
#include <QMap>
class ChunkStream class ChunkStream
{ {
public: public:
ChunkStream(const QByteArray & data) {setSource(data);} enum Version {
ChunkStream(QDataStream & str) {setSource(str);} Version_1 /*! First, old version */,
ChunkStream(QByteArray * data = 0) {setSource(data);} Version_2 /*! Second, more optimized version */ = 2,
};
ChunkStream(const QByteArray & data): version_(Version_2) {setSource(data);}
ChunkStream(QDataStream & str): version_(Version_2) {setSource(str);}
ChunkStream(QByteArray * data = 0, Version v = Version_2): version_(v) {setSource(data);}
ChunkStream(Version v): version_(v) {setSource(0);}
~ChunkStream(); ~ChunkStream();
template <typename T> template <typename T>
@@ -30,21 +36,38 @@ public:
QDataStream & dataStream() {return stream_;} QDataStream & dataStream() {return stream_;}
QByteArray data() const {return tmp_data;} QByteArray data() const {return tmp_data;}
bool atEnd() const {return stream_.atEnd();} bool atEnd() const {return stream_.atEnd();}
Version version() const {return (Version)version_;}
int read() {stream_ >> last_id >> last_data; return last_id;} int read();
void readAll();
int getID() {return last_id;} int getID() {return last_id;}
template <typename T> template <typename T>
T getData() const {T ret; QDataStream s(last_data); s.setVersion(QDataStream::Qt_4_8); s >> ret; return ret;} T getData() const {T ret; QDataStream s(last_data); s.setVersion(QDataStream::Qt_4_8); s >> ret; return ret;}
template <typename T> template <typename T>
void get(T & v) const {v = getData<T>();} void get(T & v) const {v = getData<T>();}
template <typename T>
const ChunkStream & get(int id, T & v) const {
QByteArray ba = data_map.value(id);
if (!ba.isEmpty()) {
QDataStream s(ba); s.setVersion(QDataStream::Qt_4_8);
s >> v;
}
return *this;
}
private: private:
void _init(); void _init();
static uint readVInt(QDataStream & s);
static void writeVInt(QDataStream & s, uint val);
int last_id; int last_id;
uchar version_;
QByteArray * data_, last_data, tmp_data; QByteArray * data_, last_data, tmp_data;
QBuffer buffer; QBuffer buffer;
QDataStream stream_; QDataStream stream_;
QMap<int, QByteArray> data_map;
template <typename T> friend ChunkStream & operator <<(ChunkStream & s, const ChunkStream::Chunk<T> & c);
}; };
template <typename T> template <typename T>
@@ -53,7 +76,19 @@ ChunkStream & operator <<(ChunkStream & s, const ChunkStream::Chunk<T> & c) {
QDataStream bas(&ba, QIODevice::WriteOnly); QDataStream bas(&ba, QIODevice::WriteOnly);
bas.setVersion(QDataStream::Qt_4_8); bas.setVersion(QDataStream::Qt_4_8);
bas << c.data; bas << c.data;
s.dataStream() << c.id << ba; switch (s.version_) {
case ChunkStream::Version_1:
s.stream_ << c.id << ba;
break;
case ChunkStream::Version_2:
if (s.data_->isEmpty())
s.stream_ << uchar(uchar(s.version_) | 0x80);
ChunkStream::writeVInt(s.stream_, c.id);
ChunkStream::writeVInt(s.stream_, ba.size());
s.stream_.writeRawData(ba.data(), ba.size());
break;
default: break;
}
return s; return s;
} }