消失的字符串——c语言函数中的数据存储方式以及字符串实现

2019独角兽企业重金招聘Python工程师标准>>> hot3.png

        事情是这样的,最近在学习翁老师的Mooc之C语言程序设计,插一句,真的是好老师,讲的真的太好了,在c里的字符串这里,翁老师演示了一个小程序,大概代码如下,无非就是想给我们说明c语言的scanf读入字符串是遇到空格、Tab和回车为止,并且会在结尾自动加上表示字符串结束的‘\0’,然后想给我们演示如果出现字符数组越界会怎样?!就出现了如下的示例:

#include <stdio.h>int main()
{char word[8];char word2[8];scanf("%s",word);scanf("%s",word2);printf("%s##%s\n",word,word2);return 0;
}

                            205512_AJdJ_2692223.jpg

        如上就是程序运行结果,有没有觉得很神奇,反正当时我是震惊了,WoW,太不可思议了,这究竟发生了什么,老师也留下了这个疑问给我们,留下我们苦苦思索,但是百思不得其解,天哪,还好老师给了一个小提示,这和c语言中的数据在计算机中的存放方式有关。

        于是我开始思考,我仿照翁老师的做法,反正c语言提供了这样的工具——指针,我们就输出地址来看看,我也是知道的,函数在计算机中是通过堆栈的方式来实现的,数据就是通过堆栈来保存,堆栈是很重要的数据结构,其有一个很重要的特性,FILO,后进先出。我增加了输出字符数组每个元素的地址的代码,如下:

#include <stdio.h>
int main()
{char word[8];char word2[8];scanf("%s",word);scanf("%s",word2);printf("%s##%s\n",word,word2);printf("%p\n",word);printf("%p\n",word2);printf("\n");int i;for(i=0;i<8;i++){printf("%p\n",&word[i]);}printf("\n");for(i=0;i<8;i++){printf("%p\n",&word2[i]);}return 0;
}

                                    210904_L53T_2692223.jpg

        看到了输出地地址后,加上草图,一切都简单明了了,总结如下:

        在32位的架构下(我是在32位下编译的,其实在64位下没有出现这个情况,那是因为64位下存储字大小又不同了,其实我还不太解释的明白,留一个问号???但是通过相同程序的运行,我们能够发现不同,word和word2之间的地址差为16个字节,当我将字符数组大小扩大为20时,地址差是按一个存储字增加的,那64位的存储字就应该为16字节,这也确实符合计算机组成里讲的对齐的方式存储),因为函数是通过堆栈来存储变量的,是高地址向低地址存储,我们可以把一个字符数组想成一个存储字,在字内是顺序存储的,就是从低地址向高地址,而存储字间是则是堆栈的顺序,是一种小端存放的方式,这里,word和word2看成指针是const的,是不变的,当scanf读入第一个12345678‘\0’字符串的时候是从word所指的位置向高地址依次写入,虽然越界到了规定的位置以外,但是c语言并没有提供有效的机制,即使这是不安全的,但是我们依然可以做,可以顺利的写入,然后读入第二个12345678‘\0’字符串的时候,同样的方式,从word2所指向的位置向高地址写入,很不幸的是,在定义字符数组的时候大小就决定了,并且是连续的分配的内存空间,所以写入到8的时候,其实就用完了定义的word2的数组大小,但是字符串末尾还有结束字符'\0',虽然是不安全的,但是c语言的编译器不会发现,因此'\0'被继续写到了接下来连续的地址中,就是word所指向的位置,写入就将原来的1所覆盖,读入就ok了,接下来是打印,首先先打印word所指向的字符串,第一个已经重新写入了,刚好是字符串结束字符,因此什么也不输出,然后是打印word2所指向的字符串,打印1.。。。一直要到'\0'才会认为是字符串的结束,所以就打印出了12345678‘\0',说到这里,我也就基本明白了字符串消失之谜的真相,用柯南的话,真相永远只有一个!

        哦,原来是这样,果然不是我不明白,而是我并不知道,计算机的世界真的很精彩,而且很有意思,等着我,我一定要好好看看。

213732_1RgQ_2692223.jpg

213732_WYjj_2692223.jpg

        

转载于:https://my.oschina.net/fchen24/blog/664079

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

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

相关文章

JQuery 文本框高亮显示插件

以前使用Mootools Js框架&#xff0c;文本框有个$(id).highlight()高亮方法&#xff0c;使其高亮显示&#xff0c;效果如下&#xff1a; JQuery 中没有这个方法&#xff0c;自己今天写了一个Plugin&#xff0c;把代码贴出来分享一下&#xff1b; 代码如下&#xff1a; jquery-h…

优秀工程师应该具备哪些素质_一名优秀工程师所需要具备的条件和素质[1]

一名优秀工程师所需要具备的条件和素质碰到一个不相熟的人。“上学还是工作&#xff1f;(我看起来还不算老嘛)”“工作了&#xff0c;一年多一点。”“做什么的&#xff1f;”“半导体行业的。”“哦&#xff0c;具体做什么呢&#xff1f;”“嗯......工程师。”工程师——这就…

链方法[C# 基础知识系列]专题三:如何用委托包装多个方法——委托链

最近研究链方法&#xff0c;稍微总结一下&#xff0c;以后继续补充&#xff1a; 弁言&#xff1a; 上一专题分析了下编译器是如何来翻译委托的&#xff0c;从中间语言的角度去看委托&#xff0c;希望可以帮助大家进一步的理解委托&#xff0c;然而之前的分析都是委托只是封装一…

51单片机lcd1602的简单实用

基础知识准备 lcd1602简介 图片来自b站斌哥单片机 需要关注的有以下几点&#xff1a; RS : 高数据低命令&#xff1b;RS 1时&#xff0c;lcd1602读写数据&#xff0c;RS0时&#xff0c;lcd1602读写命令R/W&#xff1a;高读低写&#xff1b;RW1时&#xff0c;单片机读取lcd160…

firedebug调试Jquery

不了解的同学先“点这里”看看什么是Firebug。简单来说&#xff0c;Firebug是Firefox上用来监视、编辑和调试站点的CSS、HTML、DOM和JavaScript的扩展工具。 我们先到Firebug的主页上下载最新的版本装上&#xff0c;下面开始调试&#xff1a; 1、查看、编辑HTML元素及其(转) 安…

linux下Bash编程until语句及格式化硬盘分区等编写脚本(十)

linux下Bash编程until语句及格式化硬盘分区等编写脚本(十)1.循环语句结构总结1.1.while语句当条件满足时&#xff0c;进入循环语句while 条件; do语句done1.2.until语句当条件不满足时&#xff0c;进入循环语句until 条件; do语句done1.3.for 变量 in 列表; do 循环体done1.4.f…

halcon区域腐蚀膨胀算子_Halcon算子

Halcon部分算子功能&#xff1a;*读取一张图像read_image(Image,C:/Users/Desktop/无标题.png)*画一个矩形生成区域draw_rectangle1(3600, Row1, Column1, Row2, Column2)*获得矩形区域gen_rectangle1(Rectangle, Row1, Column1, Row2, Column2)*区域内最大、最小、最大-最小灰…

51单片机数字钟的实现

作业 用51单片机完成一个数字钟的计时及校时功能。 1、设置8个数码管的显示缓冲区为DISPBUFF[8],并初始化为02,02,-,04,08,-,03,06. 2、定时器0每隔5ms刷新一次数码管显示缓冲区&#xff0c;并显示时间(中断来做&#xff09; 3、定时器1负责精确定时&#xff0c;计时1s,每隔…

Google Map API 学习三

转载于:https://www.cnblogs.com/MyFlora/p/3154066.html

委托声明----委托实例化-----委托调用

定义和使用委托分三个步骤&#xff1a; 1、委托声明。2、委托实例化。3、委托调用。 一、定义委托定义委托的语法如下&#xff1a; <访问修饰符> delegate 返回类型 委托名(); 定义委托和定义方法很相似&#xff0c;委托没有具体的实现体&#xff0c;由关键字delegate声明…

慎用SELECT INTO复制表

很多时候我们习惯于用SELECTINTO 复制一个表或表结构&#xff0c;因为它方便&#xff0c;快捷&#xff0c;而且在某些情况下效率比INSERT INTO 效率要高一些。但是要注意&#xff1a; SELECT INTO 复制表或表结构的时候&#xff0c;只是得到了一个“外壳”&#xff0c;就像克隆…

mysql sql语句使用技巧

mysql更新数据限制limitmysql更新时&#xff0c;要更新记录中某个区间的数据&#xff0c;只能用WHERE条件来限制了&#xff0c;用LIMIT只能限制更新多少条&#xff01;测试如下&#xff1a;UPDATE products SET goods_number goods_number10 ORDER BY goods_id DESC LIMIT 5,1…

wingdows安装psutil_psutil模块安装指南(win与linux)

1、windows下psutil模块安装&#xff1a;https://pypi.python.org/packages/3.4/p/psutil/下载符合版本的软件包下载&#xff0c;然后安装即可。2、ubuntu下载psutil模块的tar包安装复制代码 代码示例:wget https://pypi.python.org/packages/source/p/psutil/psutil-2.1.3.tar…

信号的采样与插值重建(包含matlab)

实验任务 双频信号的采样和插值重建&#xff1a;对双频信号进行采样&#xff08;符合奈奎斯特低通采样定理&#xff09;&#xff0c;绘制波形和频谱&#xff1b;编制sinc函数插值函数&#xff0c;对采样进行恢复&#xff0c;绘制波形。 实验原理 matlab代码 clear all; clos…

dc概论之IO约束

版权声明&#xff1a;转载时请以超链接形式标明文章原始出处和作者信息及本声明http://bb2hh.blogbus.com/logs/39654476.html 注明&#xff1a;如需转载&#xff0c;请注明作者出处&#xff0c;谢谢&#xff5e;&#xff0c;Author&#xff1a;pythonlong以下根据资料和个人体…

对寄存器ESP和EBP的一些理解

PS&#xff1a;EBP是当前函数的存取指针。即存储或者读取数时的指针基地址&#xff1b;ESP就是当前函数的栈顶指针。每一次发生函数的调用&#xff08;主函数调用子函数&#xff09;时&#xff0c;在被调用函数初始时&#xff0c;都会把当前函数&#xff08;主函数&#xff09;…

配置FCKeditor_2.6.3+fckeditor-java-2.4

下载 http://www.fckeditor.net/download FCKeditor.Javahttp://sourceforge.net/project/showfiles.php?group_id75348&package_id129511 -1- copy fckeditor文件夹到项目的webRoot目录下 -2- 添加Jar包 slf4j-simple-1.5.0.jar slf4j-api-1.5.0.jar java-core-2.4-beta-…

普通调幅(AM)与抑制载波双边带调幅(DSB)matlab编程实现

实验任务 实现单频信号的AM、DSB调制&#xff0c;绘制调制前后的波形和频谱 实验原理 matlab代码 AM调制 clear all; ts 0.0025; %信号抽样时间间隔 fs 1/ts; %抽样频率 t 0:ts:10-ts; %时间向量 df fs/length(t); %fft的频率分辨率 msg2 cos(2*pi*2*t); Pn fft(msg2…

electron 读取文件夹内容_如何使用Electron Framework选择,读取,保存,删除或创建文件...

本文概述为了处理文件(CRUD)的生命周期, 我们将使用对话框和文件系统组件。对话框模块提供了用于显示本机系统对话框(例如打开文件或警报)的API, 因此Web应用程序可以提供与本机应用程序和Node.js文件系统相同的用户体验。加载所需的依赖项我们需要加载以下依赖项, 以执行我们要…

MySQL中MySQL X.X Command Line Client一闪而过的问题

问题介绍&#xff1a;我安装完MySQL(我安装的是5.5)后&#xff0c;使用MySQL 5.5 Command Line Client&#xff0c;每次点击&#xff0c;总是一闪而过。解决方法如下&#xff1a;首先进入cmd 切入MySQL的安装目录&#xff0c;然后切入 bin 目录 &#xff0c;输入mysqld-nt --sk…