【C++深度剖析教程15】经典问题解析之关于string的疑问

今天来看一下在面试笔试中经常会出错的地方。
我们先来看一个代码:

#include <iostream>
#include <string>using namespace std;int main()
{string s = "12345";const char* p = s.c_str();cout << p << endl;     s.append("abced");  cout << p << endl;     return 0;
}

看一下这个代码,第一感觉是没有什么问题的,指针p指向了S这个字符串,那么当执行 s.append(“abced”);这句话之后,字符串S会增加“abced”这借个字符,那么打印输出的结果应该为:12345和12345abced。但是我们运行程序发现(纠错:使用gcc 4.4.7版本编译器),两条打印语句都是12345.这是为什么呢?下面我们看图详细分析一下:
分析
一开始只是字符串S指向0xFF112233这个内存空间,然后让P指向这个空间,执行这句话后:s.append(“abced”);S这个字符串变成了0x12345abced,同时这个字符串所对应的地址空间却变成了:0xFF445566,但是此时指针P依然指向之前的0xFF112233,里面的值是没有变化的,为什么出现这种情况呢?

因为:string对象维护了一个指向数据的char*指针,这个指针在程序运行阶段,有可能会发生突变。
所以那个P指针还是指向原来的0xFF112233这个地址,打印输出的内容就还是12345了。

纠错:

  • 针对上述的说明,有网友提出质疑,在vs2013上运行结果就是12345和12345abced。我也做了实验,在vs2017上运行结果也是12345和12345abced。而在我的Linux中,使用gcc
    4.4.7版本编译器,运行结果就是12345和12345。使用gcc 7.3.0 编译器,就得到12345和12345abced 这个结果。很明显,这与编译器实现有关,比较新版本的编译器可以得到正常的运行结果。
  • 很明显,我一开始的分析,也是有一些错误的。string对象维护了一个指向数据的char* 指针,这个指针在程序的运行阶段是有可能发生突变,也有可能不发生突变。比较新的编译器编译都没有发生突变,说明比较新的编译器解决了那个bug。
  • 在我下面的评论中,我之前说可以从另一个角度理解为什么打印结果一样,就是const,这种说法也是不对的,const修饰的变量,则该变量不能出现在赋值符号的左边,不能被直接改变,但是可以被间接改变。

下面再看一个程序:

#include <iostream>
#include <string>using namespace std;int main()
{const char* p = "12345";string s = "";s.reserve(10);  //分配内存大小为10// 不要使用 C 语言中的方式操作 C++ 中的字符串for(int i=0; i<5; i++){s[i] = p[i];}cout << s << endl;return 0;
}

这个程序运行结果为:空!!!
为什么呢?难道对S的赋值没有成功么?我们给出分析,用C语言描述C++中的字符串,会出现一些异常,看图:
这里写图片描述

这里可以看出,m_cstr是指向字符串的内容,m_length是string类的成员变量,它指向字符串的长度,经过for循环后,我们操作的不是对象的整体,m_cstr所指向的字符串确实有了,但是m_length却依然为0,所以最后打印出来的是空,我们应该直接操作这个对象,才能让m_length随着赋值而改变。

我们把程序改成这样:

#include <iostream>
#include <string>using namespace std;int main()
{const char* p = "12345";string s = "";s.reserve(10);s = p;    //直接操作对象,不要像C语言那样进行for循环赋值。// 不要使用 C 语言中的方式操作 C++ 中的字符串/*for(int i=0; i<5; i++){s[i] = p[i];}*/    cout << s << endl;return 0;
}

打印结果为:
这里写图片描述
这下就是我们期待的结果了。

总结一下:
-string类通过一个数据空间保存字符串数据
-string类通过一个成员变量保存当前字符串的长度
-C++开发时,尽量避免C语言的一些惯用的编程思想

想一起探讨以及获得各种学习资源加我:
qq:1126137994
微信:liu1126137994
可以共同交流关于嵌入式,操作系统,C++语言,C语言,数据结构等技术问题。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/423282.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

由于这台计算机没有终端服务器客户端访问许可证,远程会话被中断解决办法...

由于这台计算机没有终端服务器客户端访问许可证&#xff0c;远程会话被中断。 最近在远程连接到一台服务器突然出现这个错误&#xff0c;发现是服务器配置参数错误。安装的时候终端服务器授权模式为“每设备”&#xff0c;那么只要把终端服务器授权模式从“每设备”更改为“每用…

前端学习(83):按显示进行分类

替换元素有自己的特性&#xff0c;虽然属于inline&#xff0c;但是能改变大小

【C++深度剖析教程16】智能指针的分析

今天我们来学习C中的一个独有的特性&#xff0c;智能指针。智能指针的作用非常的强大&#xff0c;它解决了C语言关于指针部分内存泄漏的BUG。那么在此处&#xff0c;内存泄漏指的是什么呢&#xff1f; -动态申请堆空间&#xff0c;用完后不归还 -C语言中没有垃圾回收机制 -指…

【移植Linux 3.4.2内核第三步】从0制作支持新内核的文件系统

学习交流加 个人qq&#xff1a; 1126137994个人微信&#xff1a; liu1126137994学习交流资源分享qq群&#xff1a; 962535112 上一篇文章&#xff0c;我们修改了内核代码改了系统的分区&#xff0c;但是最后启动&#xff0c;发现虽然我们可以挂载之前的文件系统&#xff0c;但是…

回顾2009,展望2010

今天1月2号&#xff0c;更确切的说&#xff0c;是2010年1月2号&#xff0c;刚吃完午饭&#xff0c;突然有一股想写博客的冲动&#xff0c;不为别的&#xff0c;只为祭奠我那逝去的2009&#xff0c;再也回不来的2009。也为了能让自己在2010年有所期盼&#xff0c;有所追求&#…

javasript 操作option select

javascript select option对象总结2009-09-28 08:59一、基础理解&#xff1a;var e document.getElementById("selectId");e.options new Option("文本", "值"); //创建一个option对象&#xff0c;即在<select>标签中创建一个或多个&…

【移植Linux 3.4.2内核第二步】之修改系统分区

今天接着移植Linux 3.4.2内核&#xff0c;接着上一篇文章&#xff08;点击查看&#xff1a;上一篇文章&#xff09;我们完成了内核的串口启动打印输出&#xff0c;但是无法挂载根文件系统&#xff0c;我们看看启动后显示的是什么&#xff1a; 从打印结构可以看出&#xff0c…

phoenix Explain Plan 翻译

Explain Plan An EXPLAIN plan tells you a lot about how a query will be run: 一个执行计划会告诉你一个执行计划怎么执行 All the HBase range queries that will be executed 所有hbase 范围查询会被执行 An estimate of the number of bytes that will be scanned 评估多…

【C++深度剖析教程17】逻辑操作符的陷阱

今天我们来学习逻辑操作符&#xff0c;那么什么是逻辑操作符的陷阱呢&#xff1f;在讲解逻辑操作符陷阱之前&#xff0c;我们先来回顾什么是逻辑操作符&#xff1a; 逻辑运算符的原生语义&#xff1a; -操作数只有两种值&#xff08;true和false&#xff09; -逻辑表达式不用…

【移植Linux 3.4.2内核之四】修改内核代码支持YAFFS文件系统

上一篇文章&#xff0c;我们从0制作jffs2文件系统&#xff0c;因为我们的内核本身就支持iffs2文件系统&#xff0c;但是它不支持yaffs文件系统。今天我们就来修改内核代码&#xff0c;让内核支持yaffs文件系统的格式。 1.首先我们获取yaffs源码 我是通过git获得&#xff0c;当…

通过split命令分割大文件

场景 1.线上出了问题&#xff0c;我需要去查找log来定位问题&#xff0c;但是由于线上数据量庞大&#xff0c;这些log文件每过一个小时就会自动回滚一次&#xff0c;尽管如此&#xff0c;有的log文件依然达到了五六g以上的大小。 对于这种巨大的log文件&#xff0c;常用的一些…

一个简单的 Generic Factory 类

简单的工厂类的一个使用场景是&#xff0c; 假设有一个基类 BaseClass&#xff0c; 和一系列的子类 A&#xff0c; B&#xff0c; C&#xff0c; 工厂类根据某个参数&#xff0c;例如字符串 “A”, “B”, “C” 创建出相应的子类。 举例如下&#xff1a; public class Factory…

【移植驱动到Linux3.4.2内核之一】移植DM9000C网卡驱动

学习交流加 个人qq&#xff1a; 1126137994个人微信&#xff1a; liu1126137994学习交流资源分享qq群&#xff1a; 962535112 之前已经把uboot&#xff0c;内核&#xff0c;文件系统&#xff0c;都移植好了&#xff0c;今天开始我们把第二期写的Linux2.6.22.6内核的驱动程序全部…

位运算实现一些小算法

package com.asiainfo.cem_volte_hbase;/*** ClassName Test* Description 有意思的小算法* Author qianxl* Date 2019-08-02 17:34* Version 1.0**/ public class Test {/*** * param num1 * param num2* description: 写一个函数&#xff0c;求两个整数之和&#xff0c;要求…

关于dhtmlxScheduler的使用说明(ADD EDIT DEL,自定义CelendarBox)

由于比较多朋友询问我的使用问题&#xff0c;所以写一篇简单的使用说明&#xff1a; 可以看回以前相关的文章&#xff1a; 基于dhtmlxScheduler的个人计划... 探讨:OA系统的设计问题. weebox[ PopUp - 弹出窗 对话框 ]weebox是一个基于jquery的弹窗插件dhtmlxScheduler[ Cale…

【移植驱动到Linux3.4.2内核之二】LCD,触摸屏,按键,USB等驱动程序的移植心得总结

学习交流加 个人qq&#xff1a; 1126137994个人微信&#xff1a; liu1126137994学习交流资源分享qq群&#xff1a; 962535112 文章目录一移植驱动程序心得体会二移植LCD驱动程序记录三移植按键输入子系统驱动程序四移植触摸屏驱动程序五移植USB驱动程序今天移植了按键&#xff…

idea 解决jar 的冲突

1.maven 组件会自动去除重复的jar 注意&#xff01;灰色是去重的jar &#xff0c;实体颜色是使用的jar; 2.springboot 关于 Class path contains multiple SLF4J bindings.警告的解决 2.1查询jar的依赖结构: 查询出slf4j-log4j12 查询到然后执行exclude https://blog.csdn.ne…

永远要跟比你更成功的人在一起

永远要跟比你更成功的人在一起 (本文只有在月09日可以学习到。) 在你的朋友圈中&#xff0c;如果你是最成功的那一个&#xff0c;你就不会更成功了。 别怕与大人物打交道。最成功的人都是那些最容易与别人相处打交道的人&#xff0c;当你总是与最顶尖的人在一起时&#xff0c;你…

【C++深度剖析教程18】逗号操作符的分析

今天来学习C中的逗号操作符。那么什么是逗号操作符呢&#xff1f; *逗号操作符&#xff08;&#xff0c;&#xff09;可以构成逗号表达式 逗号表达式用于将多个子表达式连接为一个表达式逗号表达式的值为最后一个子表达式的值逗号表达式的前N-1个表达式的值可以没有返回值逗号…