String buffer using std::vector (and fixing an allocation bug).

This commit is contained in:
Ben Vanik 2014-08-06 11:37:16 -07:00
parent a085dc547e
commit 5162d69ab2
2 changed files with 28 additions and 25 deletions

View File

@ -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(); }

View File

@ -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