muduo_base代码剖析之StringPiece

本文深入解析了StringPiece类,这是由Google开发的一个高效字符串处理类,用于在网络传输中避免不必要的内存拷贝,提升性能。文章详细介绍了StringPiece的构造、成员函数及应用场景,特别适合于高性能网络通信和字符串频繁操作的场景。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

为什么要有这个StringPiece类?

  1. StringPiece类是google提供的一个类。在网络传输中,如果传递一个很大的char*类型的字符串,那么这个很大的内存块需要在网络上进行传输;这么一块字符串虽然小,但是对于网络上高性能传输上,小小的因素都将会带来网络负载的加重,因此,使用StringPiece类对string/char*进行封装,封装成const char* ptr_ + int length,并且ptr始终是一个指针,ptr的赋值不是通过strcpy进行拷贝,而仅仅是改变ptr的指针。这样StringPiece类的大小始终都是const char* ptr_ + int length,大小为4字节+4字节。
  2. StringPiece类使用的场景:目的仅仅是读取字符串的值,所以这个类的目的是传入字符串的字面值,它内部的ptr_ 这块内存不归它所有,所以不能做任何改动。
  3. 归根结底,是处于性能的考虑,用以实现高效的字符串传递,这里既可以用const char*,也可以用std::string类型作为参数,并且不涉及内存拷贝。
#ifndef MUDUO_BASE_STRINGPIECE_H
#define MUDUO_BASE_STRINGPIECE_H

#include <string.h>
#include <iosfwd>    // for ostream forward-declaration

#include <muduo/base/Types.h>

namespace muduo
{
class StringPiece {
 private:
  const char*   ptr_;
  int           length_;

 public:
  //构造函数:让ptr_指针指向形参const char* str,不存在内存拷贝
  StringPiece()
    : ptr_(NULL), length_(0) { }
  StringPiece(const char* str)
    : ptr_(str), length_(static_cast<int>(strlen(ptr_))) { }
  StringPiece(const unsigned char* str)
    : ptr_(reinterpret_cast<const char*>(str)),
      length_(static_cast<int>(strlen(ptr_))) { }
  StringPiece(const string& str)
    : ptr_(str.data()), length_(static_cast<int>(str.size())) { }
  StringPiece(const char* offset, int len)
    : ptr_(offset), length_(len) { }

  const char* data() const { return ptr_; }
  int size() const { return length_; }
  bool empty() const { return length_ == 0; }
  //起始指针
  const char* begin() const { return ptr_; }
  //终止指针
  const char* end() const { return ptr_ + length_; }

  void clear() { ptr_ = NULL; length_ = 0; }
  void set(const char* buffer, int len) { ptr_ = buffer; length_ = len; }
  void set(const char* str) {
    ptr_ = str;
    length_ = static_cast<int>(strlen(str));
  }
  void set(const void* buffer, int len) {
    ptr_ = reinterpret_cast<const char*>(buffer);
    length_ = len;
  }

  char operator[](int i) const { return ptr_[i]; }

  //移除ptr_的前n个字节
  void remove_prefix(int n) {
    ptr_ += n;
    length_ -= n;
  }
  //移除ptr_的后n个字节
  void remove_suffix(int n) {
    length_ -= n;
  }

  bool operator==(const StringPiece& x) const {
    return ((length_ == x.length_) &&
            (memcmp(ptr_, x.ptr_, length_) == 0));
  }
  bool operator!=(const StringPiece& x) const {
    return !(*this == x);
  }

#define STRINGPIECE_BINARY_PREDICATE(cmp,auxcmp)                             \
  bool operator cmp (const StringPiece& x) const {                           \
    int r = memcmp(ptr_, x.ptr_, length_ < x.length_ ? length_ : x.length_); \
    return ((r auxcmp 0) || ((r == 0) && (length_ cmp x.length_)));          \
  }
  STRINGPIECE_BINARY_PREDICATE(<,  <);
  STRINGPIECE_BINARY_PREDICATE(<=, <);
  STRINGPIECE_BINARY_PREDICATE(>=, >);
  STRINGPIECE_BINARY_PREDICATE(>,  >);
#undef STRINGPIECE_BINARY_PREDICATE

  int compare(const StringPiece& x) const {
    int r = memcmp(ptr_, x.ptr_, length_ < x.length_ ? length_ : x.length_);
    if (r == 0) {
      if (length_ < x.length_) r = -1;
      else if (length_ > x.length_) r = +1;
    }
    return r;
  }

  string as_string() const {
    return string(data(), size());
  }

  void CopyToString(string* target) const {
    target->assign(ptr_, length_);
  }

  // Does "this" start with "x"
  bool starts_with(const StringPiece& x) const {
    return ((length_ >= x.length_) && (memcmp(ptr_, x.ptr_, x.length_) == 0));
  }
};

}  // namespace muduo

// ------------------------------------------------------------------
// Functions used to create STL containers that use StringPiece
//  Remember that a StringPiece's lifetime had better be less than
//  that of the underlying string or char*.  If it is not, then you
//  cannot safely store a StringPiece into an STL container
// ------------------------------------------------------------------

#ifdef HAVE_TYPE_TRAITS
// This makes vector<StringPiece> really fast for some STL implementations
template<> struct __type_traits<muduo::StringPiece> {
  typedef __true_type    has_trivial_default_constructor;
  typedef __true_type    has_trivial_copy_constructor;
  typedef __true_type    has_trivial_assignment_operator;
  typedef __true_type    has_trivial_destructor;
  typedef __true_type    is_POD_type;
};
#endif

// allow StringPiece to be logged
std::ostream& operator<<(std::ostream& o, const muduo::StringPiece& piece);

#endif  // MUDUO_BASE_STRINGPIECE_H
#!/bin/bash set -e # 遇到错误停止执行 # 更新软件源 sudo apt update # 安装基本依赖工具 sudo apt-get install -y wget unzip make g++ build-essential # 安装特定版本的 protoc echo "安装特定版本的 protoc..." sudo apt-get install protobuf-compiler libprotobuf-dev # 验证安装 protoc echo "验证 protoc 安装..." sudo protoc --version # 安装 Boost echo "安装 Boost..." sudo apt-get install -y libboost-dev libboost-test-dev libboost-all-dev # 安装 CMake CMAKE_TAR="cmake-3.26.4-linux-x86_64.tar.gz" # 解压 CMake echo "解压 CMake ${CMAKE_TAR}..." sudo tar -zxvf ~/${CMAKE_TAR} -C /opt/ # 创建软链接到 /usr/local/bin sudo ln -s /opt/cmake-3.26.4-linux-x86_64/bin/cmake /usr/local/bin/cmake # 验证安装 CMake echo "验证 CMake 安装..." sudo cmake --version # 解压 Muduo MUDUO_ZIP="muduo-master.zip" #echo "解压 Muduo ${MUDUO_ZIP}..." unzip -o ~/${MUDUO_ZIP} -d ~/ # 进入 muduo 目录 cd ~/muduo-master # 运行 Muduo 的 build.sh 脚本 echo "运行 build.sh 构建 Muduo..." sudo ./build.sh # 进行 muduo 库安装 echo "进行 muduo 库安装... ./build.sh install" sudo ./build.sh install # 将头文件和库文件移动到标准系统目录 echo "将头文件和库文件移动到标准目录..." # 将头文件移动到 /usr/include sudo mv ~/build/release-install-cpp11/include/muduo /usr/include/ # 将库文件移动到 /usr/local/lib sudo mv ~/build/release-install-cpp11/lib/* /usr/local/lib/ #如果说这里移动的位置有问题你可以从home/你的用户名/build/release-install-cpp11/lib/ # 更新系统库路径 echo "更新系统库路径..." sudo ldconfig echo "Muduo 安装完成!" 改为centos适用的格式
最新发布
03-13
以下是将您提供的脚本改为适用于 CentOS 的格式,并解释每一步的变化以及差异所在: --- ### 修改后的脚本(适用于 CentOS) ```bash #!/bin/bash set -e # 遇到错误停止执行 # 更新软件源 sudo yum check-update || sudo yum update -y # 安装基本依赖工具 sudo yum install -y wget unzip gcc-c++ make # 安装特定版本的 protoc echo "安装特定版本的 protoc..." sudo yum install -y protobuf protobuf-devel # 验证安装 protoc echo "验证 protoc 安装..." protoc --version # 安装 Boost echo "安装 Boost..." sudo yum install -y boost boost-devel # 安装 CMake CMAKE_TAR="cmake-3.26.4-linux-x86_64.tar.gz" # 下载 CMake 压缩包 if [ ! -f "$CMAKE_TAR" ]; then echo "下载 CMake ${CMAKE_TAR}..." wget https://github.com/Kitware/CMake/releases/download/v3.26.4/$CMAKE_TAR -P ~/ fi # 解压 CMake echo "解压 CMake ${CMAKE_TAR}..." sudo tar -zxvf ~/${CMAKE_TAR} -C /opt/ # 创建软链接到 /usr/local/bin sudo ln -sf /opt/cmake-3.26.4-linux-x86_64/bin/* /usr/local/bin/ # 验证安装 CMake echo "验证 CMake 安装..." cmake --version # 解压 Muduo MUDUO_ZIP="muduo-master.zip" if [ ! -f "${MUDUO_ZIP}" ]; then echo "下载 Muduo 源码..." wget https://github.com/chenshuo/muduo/archive/master.zip -O ~/${MUDUO_ZIP} fi echo "解压 Muduo ${MUDUO_ZIP}..." unzip -o ~/${MUDUO_ZIP} -d ~/ # 进入 muduo 目录 cd ~/muduo-master # 运行 Muduo 的 build.sh 脚本 echo "运行 build.sh 构建 Muduo..." ./build.sh # 进行 muduo 库安装 echo "进行 muduo 库安装... ./build.sh install" ./build.sh install # 将头文件和库文件移动到标准系统目录 echo "将头文件和库文件移动到标准目录..." # 将头文件移动到 /usr/include sudo cp -r ~/build/release-install-cpp11/include/muduo /usr/include/ # 将库文件移动到 /usr/local/lib sudo cp -r ~/build/release-install-cpp11/lib/* /usr/local/lib/ # 更新系统库路径 echo "更新系统库路径..." sudo ldconfig echo "Muduo 安装完成!" ``` --- ### 关键修改点说明 #### 1. 使用 `yum` 替代 `apt` CentOS 默认使用的是 YUM 包管理器,而不是 APT。因此需要将所有涉及 `apt-get` 和 `apt` 的命令替换成对应的 `yum` 命令。 - **原命令**: ```bash sudo apt update sudo apt-get install ... ``` - **修改后**: ```bash sudo yum check-update || sudo yum update -y sudo yum install -y ... ``` #### 2. 编译工具名称变化 在 CentOS 中,GCC 的 C++ 编译器名为 `gcc-c++`,而 Ubuntu 中则是 `g++`。此外,`build-essential` 是 Ubuntu 特有的元包,在 CentOS 中可以直接通过单独安装 GCC、G++ 和 Make 工具来满足需求。 - **原命令**: ```bash sudo apt-get install -y wget unzip make g++ build-essential ``` - **修改后**: ```bash sudo yum install -y wget unzip gcc-c++ make ``` #### 3. Protobuf 包名调整 CentOS 中的 Protobuf 开发包通常命名为 `protobuf` 和 `protobuf-devel`,而非 `libprotobuf-dev`。 - **原命令**: ```bash sudo apt-get install protobuf-compiler libprotobuf-dev ``` - **修改后**: ```bash sudo yum install -y protobuf protobuf-devel ``` #### 4. 提供 CMake 下载功能 由于 CentOS 的默认仓库可能无法提供高版本的 CMake,所以新增了对指定版本 CMake 的下载和安装支持。 - 新增部分: ```bash if [ ! -f "$CMAKE_TAR" ]; then echo "下载 CMake ${CMAKE_TAR}..." wget https://github.com/Kitware/CMake/releases/download/v3.26.4/$CMAKE_TAR -P ~/ fi ``` #### 5. 文件复制方式统一化 为避免权限问题或其他特殊情况导致失败,改用了 `cp` 命令替代 `mv` 来拷贝文件到目标位置,并增加递归选项 `-r`。 - **原命令**: ```bash sudo mv ~/build/release-install-cpp11/include/muduo /usr/include/ sudo mv ~/build/release-install-cpp11/lib/* /usr/local/lib/ ``` - **修改后**: ```bash sudo cp -r ~/build/release-install-cpp11/include/muduo /usr/include/ sudo cp -r ~/build/release-install-cpp11/lib/* /usr/local/lib/ ``` --- ###
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值