C++面试八股文:std::string是如何实现的?
某日二师兄参加XXX科技公司的C++工程师开发岗位第18面:
面试官:
std::string
用过吧?二师兄:当然用过(废话,C++程序员就没有没用过
std::string
的)。面试官:
std::string("hello")+"world"
、"hello"+std::string("world")
和std::string("hello")+std::string("world")
的结果是什么?为什么?二师兄:前者和后者的结果都是
std::string
的对象,内容是“helloworld\0
”,而中间的这个表达式无法通过编译。原因是std::string
重载了operator+(const char*)
和operator+(const std::string&)
,但是const char*
却没有重载operator+
运算符。面试官:
std::string
有两个API
,resize
和reserve
,你知道它们之间的区别吗?二师兄:
resize
对应的是size
,resize
可以改变字符串的大小。reserve
对应的是capacity
,reserve
只能改变capacity
的大小。二师兄:当
resize
传入的参数小于字符串的szie
时,多余的字符串会被截取。当reserve
传入的参数小于capacity
时,reserve
什么也不会做。二师兄:当
resize
传入的参数大于字符串的szie
时,增加的字符串会被默认初始化。当reserve
传入的参数大于capacity
时,capacity
会被扩容。面试官:好的。可以通过下标访问
std::string
实例的内容吗?二师兄:可以的,
std::string
重载了下标运算符,可以像数组一样通过下标运算取出某个字符。面试官:你知道
std::string
的at
成员方法吗?二师兄: 嗯,和下标运算功能相似,不过不用担心越界问题。可以安全的访问字符串中的字符。
面试官:既然有
at
方法了,为什么还要重载下标运算符呢?二师兄:主要是因为性能上的考量。
at
虽然保证了不会超出字符串范围(超出范围抛出异常),但是性能低于下标操作。这就是有舍有得。为了安全使用at
,为了性能使用下标操作。C++给了你多个选择,如何选择看你的需求。面试官:那你知道
std::string
是如何实现的吗?二师兄:在
string
内部维护一个指针,这个指针指向真正的字符串的位置。面试官:能简单的写一下实现代码吗?
二师兄:好的。
class string
{
public:
string():size_(0),data_(nullptr){}
explicit string(const char* c)
{
size_ = strlen(c);
data_ = (char*)malloc(size_+1);
memset(data_,0,size_+1);
memcpy(data_,c,size_);
}
size_t size() const {return size_;}
const char* c_str() const {return data_;}
private:
size_t size_;
char* data_;
};
二师兄:在实现
append
或者+=
的时候,需要把当前字符的长度加上append
的内容的长度,以此长度申请一块新内存,然后把当前字符串的内存和append
的内容考入新申请的内存中。free
掉之前data_
指向的内存,然后把data_
指针指向新申请的内存。面试官:好的。这样的实现有一些弊端。如果频繁的对一个
std::string
对象append
内容,会发生什么?二师兄:是的,因为频繁的
malloc
和free
,会有性能问题。因所以编译器在实现std::string
的时候一般会预先申请一块大的内存,这块内存的长度是capacity
,当添加的字符串的长度加上当前的字符串长度小于capacity
时,直接添加到当前的块上即可。面试官:好的。针对字符串比较少的情况,一般编译器会做一些优化,你知道如何优化的吗?
二师兄:这个好像在哪看过,不记得额。。。
面试官:好的,今天的面试结束了,请回去等通知吧。
今天二师兄的表现不错,除了最后一个问题,基本上都答上来了。让我们来看下这个问题:
针对字符串比较少的情况,一般编译器会做一些优化,你知道如何优化的吗?
我们可以看看GCC中std::string
的实现:
typedef basic_string<char> string;
_Alloc_hider _M_dataplus;
size_type _M_string_length;
enum { _S_local_capacity = 15 / sizeof(_CharT) };
union
{
_CharT _M_local_buf[_S_local_capacity + 1];
size_type _M_allocated_capacity;
};
这里的_CharT
就是char
,所以_S_local_capacity
等于15
。当字符串的长度小于等于15
时,直接存在_M_local_buf
中,而不需要在堆中申请内存。当字符串长度大于15
时,在内存中申请一块内存,这块内存的起始地址保存在_M_dataplus
中,这块内存的容量保存在_M_allocated_capacity
中,而字符串的真实长度保存在_M_string_length
中。当向字符串中添加字符时,如果添加字符的长度大于 _M_allocated_capacity - _M_string_length
,则需要resize
,否则直接追加到_M_dataplus
保存的内存块中即可。
好了,今天的面试到这里就结束了。感谢小伙伴们的耐心阅读,咱们明天继续二师兄的面试之旅!
关注我,带你21天“精通”C++!(狗头)
C++面试八股文:std::string是如何实现的?的更多相关文章
- 《面试八股文》之 Redis 16卷
微信公众号:moon聊技术 关注选择" 星标 ", 重磅干货,第一 时间送达! [如果你觉得文章对你有帮助,欢迎关注,在看,点赞,转发] 大家好,我是 moon. redis 作为 ...
- 《面试八股文》之 JVM 20卷
微信公众号:moon聊技术 关注选择" 星标 ", 重磅干货,第一 时间送达! [如果你觉得文章对你有帮助,欢迎关注,在看,点赞,转发] 大家好,我是 moon. <面试八股 ...
- QString 和std::string互转
std::string cstr; QString qstring; //****从std::string 到QString qstring = QString(QString::fromLocal8 ...
- std::string的split函数
刚刚要找个按空格分离std::string的函数, 结果发现了stackoverflow上的这个问题. 也没仔细看, 直接拿来一试, 靠, 不对啊, 怎么分离后多出个空字符串, 也就是 "a ...
- could not deduce template argument for 'const std::_Tree<_Traits> &' from 'const std::string'
VS2008, 写一个简单的demo的时候出现了这个: 1>------ Build started: Project: GetExportTable, Configuration: Relea ...
- 源码阅读笔记 - 3 std::string 与 Short String Optimization
众所周知,大部分情况下,操作一个自动(栈)变量的速度是比操作一个堆上的值的速度快的.然而,栈数组的大小是在编译时确定的(不要说 C99 的VLA,那货的 sizeof 是运行时计算的),但是堆数组的大 ...
- CString std::string相互转换
CString->std::string 例子: CString strMfc=“test“; std::string strStl; strStl=strMfc.GetBuffer(0); s ...
- 计算std:string的字节长度
如果项目本身是使用 Unicode 字符集和utf8编码,std::string的length(),size()甚至是c的strLen取到的都是字节长度了,比如三个汉字,就是9, 以上情况不满足的话, ...
- 【原】error C2679: binary '<<' : no operator found which takes a right-hand operand of type 'std::string'
今天遇到一个非常难以排查的BUG,谷歌度娘都问过了依旧无解,最后自己重新尝试之后找到解决方案: 先看一下报错信息: 1>.\lenz.cpp(2197) error C2679: binary ...
- 类型安全且自动管理内存的返回 std::string 的 sprintf 实现
在这篇博文里,我提到了一个例子,说的是使用C++实现类型安全的printf.这个例子很惊艳,但是在我写程序的时候,并非那么"迫切"地需要它出现在我的工具箱中,因为它并不比普通的pr ...
随机推荐
- 当后端人员未提供接口,前端人员该怎么测试 --mock
1.回顾 2.线上的mock http://rap2.taobao.org/ https://www.easy-mock.com/ 3.线上接口文档 Swagger https://swagger.i ...
- 企业实践 | 国产操作系统之光? 银河麒麟KylinOS-V10(SP3)高级服务器操作系统基础安装篇
[点击 关注「 全栈工程师修炼指南」公众号 ] 设为「️ 星标」带你从基础入门 到 全栈实践 再到 放弃学习! 涉及 网络安全运维.应用开发.物联网IOT.学习路径 .个人感悟 等知识分享. 希望各位 ...
- [ElasticSearch] ES集群状态由非正常状态(red)恢复为正常状态(green)的思路与实践
1 场景描述 1.1 资源与原规划 三台主机组成ES集群的规划: 集群名: xxx_elastic 172.15.3.7 es1 master 172.15.3.8 es2 (非master) 172 ...
- Linux内存和磁盘管理(入门)
关于对Linux系统的内存和磁盘的操作 内存和磁盘使用率 查看内存使用率: free.top 查看磁盘使用率: fdisk.df.du free 以KB为单位: $ free total used f ...
- 理解Linux系统: 进程
Linux内核版本: 2.6.11.12 编写代码: 创建进程 创建进程使用fork系统调用,官方文档对于fork的描述: fork() creates a new process by duplic ...
- c/c++零基础坐牢第二天
c/c++从入门到入土(2) 开始时间2023-04-13 23:02:34 结束时间2023-04-14 01:26:05 前言:如果第一天没把你劝退,恭喜你!通过今天的学习你就能半步踏进编程的大门 ...
- 我的第一个NPM包:panghu-planebattle-esm(胖虎飞机大战)使用说明
好家伙,我的包终于开发完啦 欢迎使用胖虎的飞机大战包!! 为你的主页添加色彩 这是一个有趣的网页小游戏包,使用canvas和js开发 使用ES6模块化开发 效果图如下: (觉得图片太sb的可以自己改 ...
- 开心档之MySQL 复制表
MySQL 复制表 如果我们需要完全的复制MySQL的数据表,包括表的结构,索引,默认值等. 如果仅仅使用CREATE TABLE ... SELECT命令,是无法实现的. 本章节将为大家介绍如何完整 ...
- ROS机器人摄像头寻线
ROS机器人摄像头寻线 连接小车 注意:必须在同一区域网 ssh clbrobort@clbrobort 激活树莓派主板 roslaunch clbrobot bringup.launch 开启摄像头 ...
- 18-html压缩
const { resolve } = require('path'); const HtmlWebpackPlugin = require('html-webpack-plugin'); modul ...