出现的问题

  在做编程题的的时候,遇到了一个很奇怪的错误,出问题的代码如下:

 1 #include <cstdio>
2 using namespace std;
3
4 int main() {
5 int c;
6 bool b;
7 // printf("%p %p", &c, &b); // c的地址是:66fe1c,b的地址是:66fe1b
8 scanf("%d %d", &c, &b);
9 printf("c = %d b = %d", c, b);
10
11 return 0;
12 }

这是运行的结果:

  是不是觉得很奇怪?明明给变量c和变量b输入的是10和1,可是在输出的结果中,c的值为0。这是为什么呢?其中原因涉及到内存。大概的解释是,我通过%d占位符来为bool型变量b输入值,而系统为bool型的变量分配1个字节大小的内存,又因为通过%d输入的值是占4个字节大小的,所以由于变量b的内存不够,输入的值会占用到变量c的内存并且覆盖原来存储的值。下面,我将通过变量所对应的内存中发生的变化进行详细的解释。

详细解释

  通过打印输出变量c,b的地址,我们来绘制出相应的内存条:

  注意,由于我们在定义变量c,b时没有进行初始化,系统会为变量随机分配值,所以内存条应该是有相应的十六进制的值的,但图中为了方便,表示空白省略其中相应的值。

  为了能够更好地表示出其中的错误,我们会输入比较大的数字。

  这里我们还是先为变量c输入10,之后十进制10会转变为十六进制00 00 00 0A,倒着存储到变量c的内存中:

  在为变量b输入值之前,我们需要知道,在C++中,bool型变量的字节大小为1个字节,也就是sizeof(bool) == 1,但我们在scanf函数中通过%d占位符来为变量b读取值,又知道通过占位符%d的值是占4个字节的大小的,而b变量的内存大小为1个字节,很明显,是存储不了4个字节大小的值的,所以只能通过占用变量c的内存来存储(准确来说是占用了变量c的前3个字节的内存),导致原本存储在变量c内存中的值被覆盖。

  现在,我们为b变量输入180079837,再按照上述的说法,来看看内存发生了什么变化:

  首先十进制180079837会转变为十六进制0A BB CC DD,倒着存储到变量b的内存中,又因为变量b的内存只有1个字节的大小不够存储,所以会在接着的3个字节大小内存来存储,这就会导致属于变量c的内存被占用,我们输入的新值覆盖了原本的值。

  没错,这就是问题所在。现在我们来打印输出看看他们的值:

  嗯...又是些很奇怪的值,这是怎么读出来的呢?

  首先来看变量b,虽然我们为变量b输入的值是4个字节的大小,但由于变量b的所占的字节大小为1,所以系统自会只读66fe1b这个地址存放的值,十六进制DD对应着十进制的211,所以实际上b变量所存储的值是211。

  然后是变量c,因为变量c是int型变量,占用字节大小为4,所以系统会从66fe1c这个地址开始往后读4个字节大小的内存,这4个字节的内存就存储着变量c的值,我们读的顺序应该是00 0A BB CC,对应着十进制703436,所以变量就存储着703436,并打印输出之。

  通过,上面的解析,我们现在应该明白了一开始为什么我们输入 10 1 会出现 c = 0, b = 1 的情况了,就是下图的情况。

  所以有什么解决的方法呢?首先,bool型没有专门对应的占位符。所以如果想通过scanf函数来为bool变量进行输入,可以把该变量定义为int型并通过占位符%d进行输入,又或者是通过枚举来定义。而如果是C++的话,就更简单了,用cin来对bool型变量进行输入。

常用占位符

%d:输入/输出十进制整数。

%o:输入/输出八进制整数。

%x:输入/输出十六进制整数。

%u:输入/输出无符号型整数。

%hd:输入/输出十进制短整数。

%ld:输入/输出十进制长整数。

%lld:输入/输出十进制长长整数。

%ull:输入/输出十进制无符号长长整数。

(如果要输入/输出八进制或十六进制的数,只需要把d改为o或x)

%f:输入/输出单精度浮点数。

%lf:输入输出双精度浮点数。

%e:输入/输出科学计数。

%c:输入/输出一个字符。

%s:输入/输出字符串。

%p:输入/输出地址。

参考资料

https://tieba.baidu.com/p/7250200456

《C语言程序设计现代方法-第二版》

在scanf函数中占位符使用错误而产生的一些错误的更多相关文章

  1. scanf函数中*修饰符的作用,如:%*d

    在scanf函数中,*修饰符可以跳过所在项的输入.如下: #include <stdio.h> int main() { ; printf("请输入:"); scanf ...

  2. Spring中手动增加配置文件中占位符引用的变量

    在项目中遇到一个这样的需求,项目的配置文件由外部传入,这时spring配置文件那些占位符变量该如何取值呢? 解决这个问题的做法有几种,我想到的大概有以下三种: 1.通过系统属性来实现,把外部传入的配置 ...

  3. jpa的@Query中"?"占位符的使用小坑

    今天使用@Query自定义查询语句,出现了一个错误: java.lang.IllegalArgumentException: Parameter with that position [1] did ...

  4. C/C++ scanf 函数中%s 和%c 的简单差别

    首先声明:在键盘中敲入字符后,字符会首先保存在键盘缓冲区中供scanf函数读取(scanf.getchar等函数是读取缓冲区,getch函数是读取的控制台信息,即为直接从键盘读取).另外特别注意键盘上 ...

  5. printf函数中*修饰符的作用,如:%*d

    在printf函数中,我们可以用数字修饰来控制打印的字段宽度和精度,如下(为强调视觉效果,均填充0): #include <stdio.h> int main() { ; float f= ...

  6. Hql中占位符(转)

    在新的Hibernate 4版本中,对于Hql有一点点改变,如果你还是按照以前的方式去编写HQL并且用了以下占位符的方式,就会得到一个警告. 参考资料:https://hibernate.atlass ...

  7. Jfinal数据库操作语句中占位符的使用

    占位符的优点: 1.增加SQL代码可读性 2.占位符可以预先编译,提高执行效率 3.防止SQL注入 4.用占位符的目的是绑定变量,这样可以减少数据SQL的硬解析,所以执行效率会提高不少 假设要将id从 ...

  8. sprintboot 中占位符及多环境配置

    (原) 关于springboot中多环境配置问题 1.在application.properties文件中通过 spring.profiles.active=... 选择系统所要加载的配置文件,这里的 ...

  9. Scala学习笔记(六):本地函数、头等函数、占位符和部分应用函数

    本地函数 可以在方法内定义方法,这种方法叫本地函数,本地函数可以直接访问父函数的参数 def parent(x: Int, y: Int): Unit ={ def child(y:Int) = y ...

随机推荐

  1. Gym 100803G Flipping Parentheses

    题目链接:http://codeforces.com/gym/100803/attachments/download/3816/20142015-acmicpc-asia-tokyo-regional ...

  2. JavaScript——四

    引用案例:事件只能执行一次 array里面都是事件对象 点击box3区域时,则会有事件冒泡现象,即:box3响应后,box2(比他大一节)的区域,box1(比box2大一级)相应出现响应事件现象 捕获 ...

  3. WSL2 新建dotnet core mvc项目

    我们知道dotnet sdk会有很多命令,但在我们完全不知道如何去使用哪个命令. 我们使用dotnet -h进行查看: 我们看到SDK的new命令,但是new命令又如何使用呢? 我们再次使用帮助: 可 ...

  4. LINUX - mmap()

    内存映射函数 https://blog.csdn.net/qq_33611327/article/details/81738195

  5. 注意力(Attention)与Seq2Seq的区别

    什么是注意力(Attention)? 注意力机制可看作模糊记忆的一种形式.记忆由模型的隐藏状态组成,模型选择从记忆中检索内容.深入了解注意力之前,先简要回顾Seq2Seq模型.传统的机器翻译主要基于S ...

  6. ffmpeg开发环境搭建--(linux)

    1.     下载源码: http://ffmpeg.org/download.html 2.     解压:tar –xvf ffmpeg-3.2.1.tar.bz2 3.     配置 Eg:./ ...

  7. mysql(四)------Mysql中的锁

    1. 2 MySQL InnoDB 锁的基本类型 https://dev.mysql.com/doc/refman/5.7/en/innodb-locking.html 官网把锁分成了 8 类.所以我 ...

  8. java之 javassist简单使用

    0x01.javassist介绍 什么是javassist,这个词一听起来感觉就很懵,对吧~ public void DynGenerateClass() { ClassPool pool = Cla ...

  9. sql-libs(5)

    直接只用floor报错注入  或者 update即可

  10. npm fetch All In One

    npm fetch All In One fetch for TypeScript { "compilerOptions": { "lib": ["D ...