将stl字符串缩小到小于15个字符的容量

shrinking stl string to a capacity of less than 15 characters

本文关键字:15个 字符 容量 小于 stl 字符串 缩小      更新时间:2023-10-16

考虑std::string和容量

std::string aString = "12345678901234567890";
std::cout << aString.capacity() << std::endl; // capacity is 20
aString.clear();
std::cout << "size: " << aString.size() << std::endl;
std::cout << aString.capacity() << std::endl; // capacity is 20
aString.shrink_to_fit();
std::cout << aString.capacity() << std::endl; // capacity is 15?
std::string newString;
std::cout << newString.capacity() << std::endl; // capacity is 15?

15个字符是最小容量吗?有没有办法把它缩小到字符串的实际大小?

std::string的一些实现使用"短字符串优化",该优化使用字符串本身的空间,而不是动态分配的内存来存储小字符串的内容。所以,你不能收缩字符串,因为内存是字符串本身的一部分。

这在msvc2017 basic_string实现中是不可能的,因为构造一个空的std::string将分配至少15个字符。

更多详细信息:std::string.capacity()返回_Myres,表示当前分配存储的长度

// Implementation of std::string.capacity()
_NODISCARD size_type capacity() const noexcept
{   // return current length of allocated storage
return (this->_Get_data()._Myres);
}
// Construct an empty string would call _Tidy_init() method
basic_string() _NOEXCEPT_COND(is_nothrow_default_constructible_v<_Alty>)
: _Mybase()
{   // construct empty string
_Tidy_init();
}
// Moreover, initialize _Myres by enum constant value
void _Tidy_init()
{   // initialize basic_string data members
auto& _My_data = this->_Get_data();
_My_data._Mysize = 0;
_My_data._Myres = this->_BUF_SIZE - 1;
// the _Traits::assign is last so the codegen doesn't think the char
// write can alias this
_Traits::assign(_My_data._Bx._Buf[0], _Elem());
}
// _BUF_SIZE stands for literal number 16 (std::cout<<std::string::_BUF_SIZE;)
enum
{   // length of internal buffer, [1, 16]
_BUF_SIZE = 16 / sizeof (value_type) < 1 ? 1
: 16 / sizeof (value_type)};