众所周知,大部分情况下,操作一个自动(栈)变量的速度是比操作一个堆上的值的速度快的。然而,栈数组的大小是在编译时确定的(不要说 C99 的VLA,那货的 sizeof 是运行时计算的),但是堆数组的大小在运行时确定,很自由。此外,栈空间比堆空间有限,前者只有几MB,而后者基本上就是你系统内存的大小。

正因为这样,我们想组合两者的优势,既要享受堆空间的自由,又想要在数组较小的时候使用栈空间来加快速度,并且结合两者不会产生额外的开销,这时候,我们需要Short String Optimization (SSO).

一个 std::string 一般情况下把字符串存在堆空间,这样的效果就像你使用 new char [size] 来创建堆数组一样,这样可以避免字符串过长然后爆栈,但是也相对较慢,尤其是在需要拷贝的时候。为了优化,许多 std::string 的实现在内部装了一个短长度的栈数组,就像 char [20] 一样,如果你的 string 使用小于20个元素,那么就可以通过使用它来减少堆空间操作,加快速度。

实现细节

综上所述,我们的 std::string 至少保存以下信息:

  • 一个短的栈数组
  • 一个用来指向堆内存的指针
  • 标记你的数组存在了哪
  • 一个变量保存长度

You don't pay for what you don't use

如果你这么写,那就大错特错了

class string
{
public:
// 其他成员函数
private:
char* _heap;
size_t _size;
size_t _capacity;
char _stack[16];
};

对于64位系统,指针是 8 byte,size_t 是 8 byte,如果你要存一个< 16元素的字符串,那么这个类会因为没有用的 _heap 成员额外浪费 8 byte,如果存了>= 16元素的字符串,那么这个类会浪费 16 byte。一个比较好的解决方案是利用 union,让使用情况互斥的两个元素共用相同的空间。

在 MSVC 的 std::string 中,数据结构是这样的

union _Bxty
{ // storage for small buffer or pointer to larger one
value_type _Buf[_BUF_SIZE];
pointer _Ptr;
char _Alias[_BUF_SIZE]; // to permit aliasing
} _Bx; size_type _Mysize; // current length of string
size_type _Myres; // current storage reserved for string

无视那个 _Alias,我没有在代码的任何地方发现这个东西被使用的痕迹,目测是买来 dinkumware 的代码的时候就留在那里没有改了。

经过蓝色大大的教导,应该是为了让编译器允许pointer alias,蓝色大大的原话是这样的:

允许编译器做 pointer alias,其中指向这个 union 的指针可以与指向 char 的 char* 指针做 alias。而由于 char* 指针可以与任意其他类型指针做 alias,所以允许指向这个结构体的指针与任意其他类型的指针做 alias。

_BUF_SIZE = 16 / sizeof (value_type) < 1 ? 1 : 16 / sizeof (value_type)

这里保证small_buffer元素至少有一个。

基于 MSVC 的实现自己造对象串

基于 MSVC 的实现,我们只需要提供以下东西,就能用我们自己的类,使用 std::basic_string<…> 制造一个有 SSO 支持的数组

std::basic_string 的三个模板参数,分别是类本体,它的 traits,以及 allocator,traits 里面至少要包含以下函数,分别是

static void copy(MyClass* dest, const MyClass* src, size_t count); //拷贝
static void move(MyClass* dest, MyClass* src, size_t count); //移动
static void assign(MyClass& lhs, const MyClass& rhs); //赋值
static void assign(MyClass* dest, size_t count, const MyClass& value); //赋值序列
static MyClass* find(MyClass* from, size_t count, const MyClass& value); //在[from, from + count)内查找value
static int compare(MyClass* ptr, MyClass* pother, size_t count); //比较
static size_t length(MyClass* ptr); //计算长度

如果有必要,你也可以提供接受右值引用的版本。

值得一提的是你的类也需要有类似于 C 字符串用来标记结尾的特殊值,以及不能大于 16byte,否则 small buffer 里面只有一个元素,没有意义了。

然后就可以用啦。

源码阅读笔记 - 3 std::string 与 Short String Optimization的更多相关文章

  1. 源码阅读笔记 - 2 std::vector (1)

    vector的源码真是太长了,今天用了一个下午和一个晚上看和注释了前面的一千行左右 p.s.博客园的代码高亮真是太垃圾, 如果想要阅读带注释的源码,推荐粘贴到VS2015里,然后按ctrl+z取消自动 ...

  2. 源码阅读笔记 - 2 std::vector (2) 关于Allocator Aware Container特性

    所有的STL容器,都保存一个或默认,或由用户提供的allocator的实例,用来提供对象内存分配和构造的方法(除了std::array),这样的容器,被称作Allocator Aware Contai ...

  3. 源码阅读笔记 - 1 MSVC2015中的std::sort

    大约寒假开始的时候我就已经把std::sort的源码阅读完毕并理解其中的做法了,到了寒假结尾,姑且把它写出来 这是我的第一篇源码阅读笔记,以后会发更多的,包括算法和库实现,源码会按照我自己的代码风格格 ...

  4. mxnet源码阅读笔记之include

    写在前面 mxnet代码的规范性比Caffe2要好,看起来核心代码量也小很多,但由于对dmlc其它库的依赖太强,代码的独立性并不好.依赖的第三方库包括: cub dlpack dmlc-core go ...

  5. CI框架源码阅读笔记5 基准测试 BenchMark.php

    上一篇博客(CI框架源码阅读笔记4 引导文件CodeIgniter.php)中,我们已经看到:CI中核心流程的核心功能都是由不同的组件来完成的.这些组件类似于一个一个单独的模块,不同的模块完成不同的功 ...

  6. CI框架源码阅读笔记3 全局函数Common.php

    从本篇开始,将深入CI框架的内部,一步步去探索这个框架的实现.结构和设计. Common.php文件定义了一系列的全局函数(一般来说,全局函数具有最高的加载优先权,因此大多数的框架中BootStrap ...

  7. PHP源码阅读笔记一(explode和implode函数分析)

    PHP源码阅读笔记一一.explode和implode函数array explode ( string separator, string string [, int limit] )此函数返回由字符 ...

  8. Apollo源码阅读笔记(二)

    Apollo源码阅读笔记(二) 前面 分析了apollo配置设置到Spring的environment的过程,此文继续PropertySourcesProcessor.postProcessBeanF ...

  9. Apollo源码阅读笔记(一)

    Apollo源码阅读笔记(一) 先来一张官方客户端设计图,方便我们了解客户端的整体思路. 我们在使用Apollo的时候,需要标记@EnableApolloConfig来告诉程序开启apollo配置,所 ...

随机推荐

  1. your local repository contains non-ascii

    安装CCS时候遇到  your local repository contains non-ascii 问题. 解决方法:  不要在中文目录下安装.

  2. iredmail安装脚本分析(一)---iRedmail.sh

    iredmail是一套以postfix为核心的整合邮件系统的安装脚本,可以达到快速部署邮件服务器的目的.为了让自己不遗忘shell的语法,所以闲来无事,学习一下他的代码. 我从官网下载他的最新版,解压 ...

  3. Nothing about semantics

    Motivation fork a project in github, seriously. Candidates PasaLab / cichlid 80% Distributed RDFS &a ...

  4. Java中的List操作

    1. 数组转List String[] arr={"1","2","3"}; List<String> list = Array ...

  5. 拓展Yii Framework(易框架)

    1.拓展yii 此文针对Yii1.1.15而写,请注意甄别你的Yii Framework 版本. 拓展yii是开发期间常见的代码处理方式.例如,你写一个新的controller(业务控制器),你通过继 ...

  6. php面试题及答案

    1.用PHP打印出前一天的时间,格式是2006-5-10 22:21:21 <?php   //echo date('Y-m-d H:i:s',time()-60*60*24   echo da ...

  7. 《java编程思想》读书笔记 暂停一段时间,改为上面的练习题

    发现个很尴尬的现象.我一天实在看得太快了...全写下 写博客都得一晚上.. 之前因为是第一次看这么厚的书,别人都说很难,以为会看很慢的.然而,已经完全学过Java的 我感觉没啥压力,越看越快....第 ...

  8. unity官方换装教程Character Customization 学习笔记

    1. 下载示例demo,可以直接从AssetsStore上下载,但是速度比较慢,我在github上找了一个据说支持unity5.x的. 链接:https://github.com/spacebeagl ...

  9. HttpServletRequest常用获取URL的方法

    1.request.getRequestURL() 返回的是完整的url,包括Http协议,端口号,servlet名字和映射路径,但它不包含请求参数.2.request.getRequestURI() ...

  10. oracle单机改变归档路径

    oracle 归档日志文件路径设置 1.  查看LOG_ARCHIVE_DEST 与 ( LOG_ARCHIVE_DEST_n 或 DB_RECOVERY_FILE_DEST )参数情况注意(  LO ...