scanf_s
scanf_s()函数是Microsoft公司VS开发工具提供的一个功能相同的安全标准输入函数,从vc++2005开始,VS系统提供了scanf_s()。在调用该函数时,可以提供一个数字以表明最多读取多少位字符。
- MSDN上scanf_s的简单模型:
int scanf_s( const char *format [, argument]...);
- 一般我们常用的格式:
scanf_s(输入格式,输入流 [,输入变量的大小(单位:字节)])
- 返回值:int类型(正常读取返回1;读取异常返回0)
使用scanf_s
- 一个字符输入
int main()
{
char n = 0;
scanf_s("%c", &n, sizeof(n));
printf("%c\n", n);
return 0;
}
2. 一个整型输入
int main()
{
int n = 0;
scanf_s("%d", &n, sizeof(n));
printf("%d\n", n);
return 0;
}
3. 多个整形输入
int main()
{
int n = 0;
int m = 0;
scanf_s("%d%d", &n, &m, sizeof(n), sizeof(m));
printf("n = %d,m = %d\n", n, m);
return 0;
}
注意:在scanf_s中多个输入的格式问题
- 一个字符数组输入
这里用%s输入一个字符数组时,最后的‘\0’是需要占1个字节的空间的;而当输入的长度超过了数组的长度时,scanf_s的安全检测就体现了,它直接返回0;printf也不会打印出来,只有一个空行,没有报错
int main()
{
char arr[5] = { 0 };
scanf_s("%s", arr, sizeof(arr));
printf("%s\n", arr);
return 0;
}
scanf
这是C语言标准库里面的一个函数,在VS底下使用这个会有警告,我们在文件开头加上:(#define _CRT_SECURE_NO_WARNINGS 1)或者 (#pragma warning(disable:4996)
就不会有警告了;参考文章:scanf在VS底下不安全的解决方法
- MSDN上scanf的简单模型:
int scanf( const char *format [,argument]... );
- 一般我们常用的格式:
scanf(输入格式,输入流 )
- 返回值:int类型(正常读取返回读取成功的元素个数;读取异常返回EOF)
使用scanf
注意:scanf里面格式化输入,不能设定域宽,printf格式化输出可以设定的
- 一个整数输入
- 多个整数输入
- 字符数组输入
可以看到scanf在输入字符串的时候,并没有安全检测,而是会直接用了数组后面的栈空间,然后造成栈溢出的问题
特例:
- scanf中含有 ‘\n’
int main()
{
char password[20] = { 0 };
scanf("%s\n", password);
for (int i = 0; i < 20; i++)
{
printf("%c ", password[i]);
}
return 0;
}
为什么回车后还要输入字符回车才结束呢?
scanf比较笨拙,要求用户输入和它第一个参数一模一样的内容.刚好\n和回车的效果是一样的,于是你第一次回车的时候它去取用户输入,发现回车还没输入(就是被scanf当做取字符分析而不是当做\n处理了).于是要再敲一次.
- scanf中含有空白字符
针对scanf输入字符的时候
int main()
{
char n = 0;
while (~(scanf(" %c",&n)))
{
printf("打印的:%c\n", n);
}
}
带空格的结果
输入不带空格的结果
在这里我们可以看到,scanf输入前加上
空格
会使我们在多组输入字符
的时候避免读到回车,对结果造成影响 ,在一些OJ题目中,经常需要这样做(也可以用getchar()来避免的)
- scanf带 * 的格式化输入
可以看到让y的值不是2而变成了3,就是跳过一个缓存里面的值吧。
总结
- 使用scanf_s函数虽然安全性比较高,但是可移植性差。scanf_s只能用在VS编译器上,而其他的用不了它
- scanf函数虽然没有scanf_s函数那么安全,但可移植性是比scanf_s强