1. 程式人生 > >二,leveldb原始碼分析(status)

二,leveldb原始碼分析(status)

// Copyright (c) 2011 The LevelDB Authors. All rights reserved.// Use of this source code is governed by a BSD-style license that can be// found in the LICENSE file. See the AUTHORS file for names of contributors.//// A Status encapsulates the result of an operation. It may indicate success,// or it may indicate an error with an associated error message.//// Multiple threads can invoke const methods on a Status without// external synchronization, but if any of the threads may call a// non-const method, all threads accessing the same Status must use// external synchronization.#ifndef STORAGE_LEVELDB_INCLUDE_STATUS_H_#define STORAGE_LEVELDB_INCLUDE_STATUS_H_#include #include "leveldb/slice.h"namespace leveldb {class Status { public: // 建立一個status物件預設建構函式 Status() : state_(NULL) { } //解構函式 ~Status() { delete[] state_; } // 拷貝建構函式和賦值運算子 Status(const Status& s); void operator=(const Status& s); // 返回 一個成功的狀態 static Status OK() { return Status(); } //不同的狀態 // Return error status of an appropriate type. static Status NotFound(const Slice& msg, const Slice& msg2 = Slice()) { return Status(kNotFound, msg, msg2); } static Status Corruption(const Slice& msg, const Slice& msg2 = Slice()) { return Status(kCorruption, msg, msg2); } static Status NotSupported(const Slice& msg, const Slice& msg2 = Slice()) { return Status(kNotSupported, msg, msg2); } static Status InvalidArgument(const Slice& msg, const Slice& msg2 = Slice()) { return Status(kInvalidArgument, msg, msg2); } static Status IOError(const Slice& msg, const Slice& msg2 = Slice()) { return Status(kIOError, msg, msg2); } // NULL 就是OK的狀態 各種狀態檢查 bool ok() const { return (state_ == NULL); } // Returns true iff the status indicates a NotFound error. bool IsNotFound() const { return code() == kNotFound; } // Returns true iff the status indicates a Corruption error. bool IsCorruption() const { return code() == kCorruption; } // Returns true iff the status indicates an IOError. bool IsIOError() const { return code() == kIOError; } // Returns true iff the status indicates a NotSupportedError. bool IsNotSupportedError() const { return code() == kNotSupported; } // Returns true iff the status indicates an InvalidArgument. bool IsInvalidArgument() const { return code() == kInvalidArgument; } // Return a string representation of this status suitable for printing. // Returns the string "OK" for success. std::string ToString() const; private: // OK status has a NULL state_. Otherwise, state_ is a new[] array // of the following form: // state_[0..3] == length of message // state_[4] == code // state_[5..] == message //把狀態訊息長度,code碼,訊息內容,封裝到message裡面 0-3 是訊息長度 4是code碼 5以後是訊息內容 const char* state_; //列舉的5個狀態 enum Code { kOk = 0, kNotFound = 1, kCorruption = 2, kNotSupported = 3, kInvalidArgument = 4, kIOError = 5 }; //由於第四個位元組是錯誤碼,返回第一個位元組 Code code() const { return (state_ == NULL) ? kOk : static_cast(state_[4]); } //複製一個status Status(Code code, const Slice& msg, const Slice& msg2); static const char* CopyState(const char* s);};inline Status::Status(const Status& s) { state_ = (s.state_ == NULL) ? NULL : CopyState(s.state_);}inline void Status::operator=(const Status& s) { // The following condition catches both aliasing (when this == &s), // and the common case where both s and *this are ok. if (state_ != s.state_) { delete[] state_; state_ = (s.state_ == NULL) ? NULL : CopyState(s.state_); }}} // namespace leveldb#endif // STORAGE_LEVELDB_INCLUDE_STATUS_H_// Copyright (c) 2011 The LevelDB Authors. All rights reserved.// Use of this source code is governed by a BSD-style license that can be// found in the LICENSE file. See the AUTHORS file for names of contributors.#include #include "port/port.h"#include "leveldb/status.h"namespace leveldb {//狀態copyconst char* Status::CopyState(const char* state) { uint32_t size; memcpy(&size, state, sizeof(size));//從state中取出4個位元組給size char* result = new char[size + 5];//+5是因為包含了前面訊息長度(4個位元組),和code碼一個位元組 memcpy(result, state, size + 5);// return result;}Status::Status(Code code, const Slice& msg, const Slice& msg2) { assert(code != kOk); const uint32_t len1 = msg.size();//msg的長度 const uint32_t len2 = msg2.size();//msg2的長度 const uint32_t size = len1 + (len2 ? (2 + len2) : 0); char* result = new char[size + 5]; memcpy(result, &size, sizeof(size)); result[4] = static_cast(code); memcpy(result + 5, msg.data(), len1); if (len2) { result[5 + len1] = ':'; result[6 + len1] = ' '; memcpy(result + 7 + len1, msg2.data(), len2); } state_ = result;}std::string Status::ToString() const { if (state_ == NULL) { return "OK"; } else { char tmp[30]; const char* type; switch (code()) { case kOk: type = "OK"; break; case kNotFound: type = "NotFound: "; break; case kCorruption: type = "Corruption: "; break; case kNotSupported: type = "Not implemented: "; break; case kInvalidArgument: type = "Invalid argument: "; break; case kIOError: type = "IO error: "; break; default: snprintf(tmp, sizeof(tmp), "Unknown code(%d): ", static_cast(code())); type = tmp; break; } std::string result(type); uint32_t length; memcpy(&length, state_, sizeof(length)); result.append(state_ + 5, length); return result; }}} // namespace leveldb