Commit 227f5881 authored by zhujiashun's avatar zhujiashun

redis_server_protocol: add append_decimal to IOBufappender

parent 7ba68500
...@@ -40,8 +40,6 @@ const char* RedisReplyTypeToString(RedisReplyType type) { ...@@ -40,8 +40,6 @@ const char* RedisReplyTypeToString(RedisReplyType type) {
} }
bool RedisReply::SerializeTo(butil::IOBufAppender* appender) { bool RedisReply::SerializeTo(butil::IOBufAppender* appender) {
char prefix_buf[24]; // should be enough for '<type><integer>\r\n"
size_t len = 0;
switch (_type) { switch (_type) {
case REDIS_REPLY_ERROR: case REDIS_REPLY_ERROR:
// fall through // fall through
...@@ -55,42 +53,34 @@ bool RedisReply::SerializeTo(butil::IOBufAppender* appender) { ...@@ -55,42 +53,34 @@ bool RedisReply::SerializeTo(butil::IOBufAppender* appender) {
appender->append("\r\n"); appender->append("\r\n");
break; break;
case REDIS_REPLY_INTEGER: case REDIS_REPLY_INTEGER:
prefix_buf[0] = ':'; appender->push_back(':');
len = butil::AppendDecimal(&prefix_buf[1], _data.integer); appender->append_decimal(_data.integer);
prefix_buf[len + 1] = '\r'; appender->append("\r\n");
prefix_buf[len + 2] = '\n';
appender->append(prefix_buf, len + 3 /* 1 for ':', 2 for "\r\n" */);
break; break;
case REDIS_REPLY_STRING: case REDIS_REPLY_STRING:
prefix_buf[0] = '$'; appender->push_back('$');
len = butil::AppendDecimal(&prefix_buf[1], _length); appender->append_decimal(_length);
prefix_buf[len + 1] = '\r'; appender->append("\r\n");
prefix_buf[len + 2] = '\n'; if (_length != npos) {
appender->append(prefix_buf, len + 3 /* 1 for ':', 2 for "\r\n" */);
if (_length == npos) {
break;
}
if (_length < (int)sizeof(_data.short_str)) { if (_length < (int)sizeof(_data.short_str)) {
appender->append(_data.short_str, _length); appender->append(_data.short_str, _length);
} else { } else {
appender->append(_data.long_str, _length); appender->append(_data.long_str, _length);
} }
appender->append("\r\n"); appender->append("\r\n");
}
break; break;
case REDIS_REPLY_ARRAY: case REDIS_REPLY_ARRAY:
prefix_buf[0] = '*'; appender->push_back('*');
len = butil::AppendDecimal(&prefix_buf[1], _length); appender->append_decimal(_length);
prefix_buf[len + 1] = '\r'; appender->append("\r\n");
prefix_buf[len + 2] = '\n'; if (_length != npos) {
appender->append(prefix_buf, len + 3 /* 1 for ':', 2 for "\r\n" */);
if (_length == npos) {
break;
}
for (int i = 0; i < _length; ++i) { for (int i = 0; i < _length; ++i) {
if (!_data.array.replies[i].SerializeTo(appender)) { if (!_data.array.replies[i].SerializeTo(appender)) {
return false; return false;
} }
} }
}
break; break;
case REDIS_REPLY_NIL: case REDIS_REPLY_NIL:
appender->append("$-1\r\n"); appender->append("$-1\r\n");
......
...@@ -670,6 +670,11 @@ public: ...@@ -670,6 +670,11 @@ public:
int append(const void* data, size_t n); int append(const void* data, size_t n);
int append(const butil::StringPiece& str); int append(const butil::StringPiece& str);
// Format integer |d| to back side of the internal buffer, which is much faster
// than snprintf(..., "%lu", d).
// Returns 0 on success, -1 otherwise.
int append_decimal(long d);
// Push the character to back side of the internal buffer. // Push the character to back side of the internal buffer.
// Costs ~3ns while IOBuf.push_back costs ~13ns on Intel(R) Xeon(R) CPU // Costs ~3ns while IOBuf.push_back costs ~13ns on Intel(R) Xeon(R) CPU
// E5-2620 @ 2.00GHz // E5-2620 @ 2.00GHz
......
...@@ -285,6 +285,25 @@ inline int IOBufAppender::append(const StringPiece& str) { ...@@ -285,6 +285,25 @@ inline int IOBufAppender::append(const StringPiece& str) {
return append(str.data(), str.size()); return append(str.data(), str.size());
} }
inline int IOBufAppender::append_decimal(long d) {
char buf[24]; // enough for decimal 64-bit integers
size_t n = sizeof(buf);
bool negative = false;
if (d < 0) {
negative = true;
d = -d;
}
do {
const long q = d / 10;
buf[--n] = d - q * 10 + '0';
d = q;
} while (d);
if (negative) {
buf[--n] = '-';
}
return append(buf + n, sizeof(buf) - n);
}
inline int IOBufAppender::push_back(char c) { inline int IOBufAppender::push_back(char c) {
if (_data == _data_end) { if (_data == _data_end) {
if (add_block() != 0) { if (add_block() != 0) {
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment