String buffer using std::vector (and fixing an allocation bug).
This commit is contained in:
parent
a085dc547e
commit
5162d69ab2
|
@ -9,17 +9,26 @@
|
||||||
|
|
||||||
#include <alloy/string_buffer.h>
|
#include <alloy/string_buffer.h>
|
||||||
|
|
||||||
|
#include <algorithm>
|
||||||
|
|
||||||
namespace alloy {
|
namespace alloy {
|
||||||
|
|
||||||
StringBuffer::StringBuffer(size_t initial_capacity) : offset_(0) {
|
StringBuffer::StringBuffer(size_t initial_capacity) {
|
||||||
buffer_.resize(MAX(initial_capacity, 1024));
|
buffer_.reserve(MAX(initial_capacity, 1024));
|
||||||
}
|
}
|
||||||
|
|
||||||
StringBuffer::~StringBuffer() = default;
|
StringBuffer::~StringBuffer() = default;
|
||||||
|
|
||||||
void StringBuffer::Reset() {
|
void StringBuffer::Reset() { buffer_.resize(0); }
|
||||||
offset_ = 0;
|
|
||||||
buffer_[0] = 0;
|
void StringBuffer::Grow(size_t additional_length) {
|
||||||
|
size_t old_capacity = buffer_.capacity();
|
||||||
|
if (buffer_.size() + additional_length <= old_capacity) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
size_t new_capacity =
|
||||||
|
std::max(buffer_.size() + additional_length, old_capacity * 2);
|
||||||
|
buffer_.reserve(new_capacity);
|
||||||
}
|
}
|
||||||
|
|
||||||
void StringBuffer::Append(const std::string& value) {
|
void StringBuffer::Append(const std::string& value) {
|
||||||
|
@ -34,27 +43,20 @@ void StringBuffer::Append(const char* format, ...) {
|
||||||
}
|
}
|
||||||
|
|
||||||
void StringBuffer::AppendVarargs(const char* format, va_list args) {
|
void StringBuffer::AppendVarargs(const char* format, va_list args) {
|
||||||
while (true) {
|
int length = vsnprintf(nullptr, 0, format, args);
|
||||||
int len = vsnprintf(buffer_.data() + offset_, buffer_.size() - offset_ - 1,
|
auto offset = buffer_.size();
|
||||||
format, args);
|
Grow(length + 1);
|
||||||
if (len == -1) {
|
buffer_.resize(buffer_.size() + length);
|
||||||
buffer_.resize(buffer_.size() * 2);
|
vsnprintf(buffer_.data() + offset, buffer_.capacity() - 1, format, args);
|
||||||
continue;
|
buffer_[buffer_.size()] = 0;
|
||||||
} else {
|
|
||||||
offset_ += len;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
buffer_[offset_] = 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void StringBuffer::AppendBytes(const uint8_t* buffer, size_t length) {
|
void StringBuffer::AppendBytes(const uint8_t* buffer, size_t length) {
|
||||||
if (offset_ + length > buffer_.size()) {
|
auto offset = buffer_.size();
|
||||||
buffer_.resize(MAX(buffer_.size() * 2, buffer_.size() + length));
|
Grow(length + 1);
|
||||||
}
|
buffer_.resize(buffer_.size() + length);
|
||||||
memcpy(buffer_.data() + offset_, buffer, length);
|
memcpy(buffer_.data() + offset, buffer, length);
|
||||||
offset_ += length;
|
buffer_[buffer_.size()] = 0;
|
||||||
buffer_[offset_] = 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const char* StringBuffer::GetString() const { return buffer_.data(); }
|
const char* StringBuffer::GetString() const { return buffer_.data(); }
|
||||||
|
|
|
@ -22,7 +22,7 @@ class StringBuffer {
|
||||||
StringBuffer(size_t initial_capacity = 0);
|
StringBuffer(size_t initial_capacity = 0);
|
||||||
~StringBuffer();
|
~StringBuffer();
|
||||||
|
|
||||||
size_t length() const { return offset_; }
|
size_t length() const { return buffer_.size(); }
|
||||||
|
|
||||||
void Reset();
|
void Reset();
|
||||||
|
|
||||||
|
@ -36,8 +36,9 @@ class StringBuffer {
|
||||||
char* EncodeBase64();
|
char* EncodeBase64();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
void Grow(size_t additional_length);
|
||||||
|
|
||||||
std::vector<char> buffer_;
|
std::vector<char> buffer_;
|
||||||
size_t offset_;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace alloy
|
} // namespace alloy
|
||||||
|
|
Loading…
Reference in New Issue