数据类型中的sizeof操作符与scanf函数
一.sizeof操作符
一些要点:
1.sizeof 操作符的操作数可以是类型,也可是变量或者表达式。
2.sizeof 的操作数如果不是类型,是表达式的时候,可以省略掉后边的括号的。
3.sizeof 后边的表达式是不真实参与运算的,根据表达式的类型来得出大小。
例:
#include <stdio.h>
int main()
{
int a = 1;
int b = 2;
int c = 3;
printf("%zd\n", sizeof(b = a + c));
printf("%d\n", b);
return 0;
}
运行结果
所以,b还是2,这是因为sizeof 在代码进行编译的时候,就根据表达式的类型确定了返回值,而表达式的执行却要在程序运行期间
才能执行,在编译期间已经将sizeof处理掉了,所以在运行期间就不会执行表达式了。
4.sizeof 的计算结果是 size_t 类型的。
二.printf可以定制占位符的输出格式
1.限定宽度
1.
例: %5d 表示这个占位符的宽度至少为5位。
如果不满5位,对应的值的前面会添加空格。
printf("%-5d\n", 123);
// 输出为 " 123 "
2.
对于小数,这个限定符会限制所有数字的最小显示宽度。
printf("%12f\n", 123.45);
// 输出 " 123.450000"
2.总是显示正负号
例:默认情况下, printf() 不对正数显示 + 号,只对负数显示 - 号。
如果想让正数也输出 + 号,可以在占位符的 % 后⾯加⼀个 + 。
输出的值默认是右对齐,即输出内容前面会有空格;
如果希望改成左对齐,在输出内容后面添加空格,
可以在占位符的 % 的后面插⼊⼀个 - 号。
printf("%+d\n", 12); // 输出 +12
printf("%+d\n", -12); // 输出 -12
3.限定小数位数
例:希望⼩数点后⾯只保留两位,占位符可以写成 %.2f 。
printf("Number is %.2f\n", 0.5);
// 输出 Number is 0.50
4.输出部分字符串
例:⽤用%.[m]s 指定输出字符串的长度,
其中 [m] 代表⼀个数字,表示所要输出的长度。
printf("%6.2f\n", 0.5);
// 输出为 " 0.50"
三. scanf
1.它的第⼀个参数是⼀个格式字符串,
里面会放置占位符(与 printf() 的占位符基本⼀致),
告诉编译器如何解读用户的输入 ,需要提取的数据是什么类型。
2.它的其余参数就是存放用户输入的变量,
格式字符串里面有多少个占位符,就有多少个变量。
3.scanf() 处理数值占位符时,
会自动过滤空白字符,包括空格、制表符、换行符等。
所以,用户输入的数据之间,有一个或多个空格不影响 scanf() 解读数据。另外,用户使用回车键,将输入分成几行,也不影响解读。
4.scanf() 处理用户输入的原理是,用户的输入先放入缓存,
等到按下回车键后,按照占位符对缓存进行解读。
解读用户输入时,会从上一次解读遗留的第一个字符开始,
直到读完缓存,或者遇到第⼀个不符合条件的字符为止。
3.1返回值
1.返回值是⼀个整数,表示成功读取的变量个数。
2.如果没有读取任何项,或者匹配失败,则返回 0 。
3.如果在成功读取任何数据之前,发⽣了读取错误或者遇到读取到⽂件结尾,则返回常量 EOF (-1)。
EOF - end of file ⽂件结束标志
3.2占位符
3.2.1
%c 不忽略空白字符,
总是返回当前第⼀个字符,无论该字符是否为空格。
3.2.2
%s ,它其实不能简单地等同于字符串。
它的规则是,从当前第⼀个非空白字符开始读起,
直到遇到空白字符(即空格、换行符、制表符等)为止。
3.2.3
%s 不会包含空白字符,所以无法用来读取多个单词,
除非多个 %s ⼀起使⽤。这也意味着,
scanf() 不适合读取可能包含空格的字符串,比如书名或歌曲名。
3.2.4
另外,
scanf() 遇到 %s 占位符,会在字符串变量末尾存储⼀个空字符 \0 。
3.2.5
scanf() 将字符串读⼊字符数组时,
不会检测字符串是否超过了数组长度。
所以,储存字符串时,很可能会超过数组的边界,
导致预想不到的结果。
3.2.6
为了防⽌这种情况,使⽤ %s 占位符时,
应该指定读入字符串的最长长度,即写成 %[m]s ,
其中的 [m] 是⼀个整数,
表示读取字符串的最大长度,
后⾯的字符将被丢弃。
3.2.7
scanf函数在C语言中被认为存在一定的安全隐患,主要是因为以下几个原因:
3.2.7.1缓冲区溢出:
scanf会将用户输入的数据直接复制到程序指定的位置,
如果没有正确地限制输入的长度,
如果用户输入的数据超过预期,
就会覆盖相邻的内存区域,
可能导致数据破坏或泄露,甚至引发程序崩溃。
3.2.7.2格式化字符串的安全性:
scanf依赖于一个格式化字符串,
它描述了如何解析输入。
如果这个字符串被恶意修改,
可能会导致解析错误或者执行未授权的操作,
例如注入格式化字符串攻击(Format String Attack)。
3.2.7.3未经验证的输入:
scanf假定所有接收到的数据都是有效的,
但在实际应用中,很难完全保证用户输入的合法性,
比如可能存在空字符、控制字符等,这可能导致意外的行为。
3.2.7.4宽字符支持有限:
对于宽字符(wchar_t),
scanf处理起来不够友好,容易引起混乱。
为了提高安全性,建议避免直接使用scanf
获取用户输入,而是考虑使用更安全的替代方法如fgets配合sscanf,
或者使用C++标准库中的std::cin,
以及对输入进行适当的验证和清理。
同时,使用strncpy而不是直接复制字符串也能减少风险。
三.赋值忽略符
有时,用户的输入可能不符合预定的格式。
为了避免这种情况,
scanf() 提供了⼀个赋值忽略符(assignment suppression character) * 。
只要把 * 加在任何占位符的百分号后面,该占位符就不会返回值,
解析后将被丢弃。
总结
今天学习了sizeof操作符,printf函数,scanf函数
对sizeof操作符,printf函数,scanf函数的分享就到这里了,如果感觉不错,希望可以给博主点个赞或者关注,感谢大家的支持,成为我继续分享的动力,还有什么问题和建议可以在评论区评论,拜拜。