scanf与scanf_s函数的使用 详解

本文详细介绍了C语言中的安全输入函数scanf_s与标准输入函数scanf的区别和用法。scanf_s在安全性上有优势,能防止缓冲区溢出,但可能不适用于所有编译器。而scanf虽然可能存在安全隐患,但在可移植性上更胜一筹。通过多个代码示例,解释了两者的参数使用、返回值和安全性检查,并提醒在使用scanf时需注意格式符间空格的影响。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

1.scanf_s(是vs提供的函数)

a.代码1

int main()
{
	char a = 0;
	//scanf_s("%c", &a, 1);
	scanf_s("%c", &a, sizeof(a));
	return 0;
}

scanf_s有三个参数,最后一个是变量a所占据空间的大小(单位为字节),这里可以写1,也可以写sizeof(a)。如果a为整型的话,第三个参数就为4,或者sizeof(a)。

b.代码2

int main()
{
	char a = 0, b = 0;
	scanf_s("%c %c", &a, sizeof(a), &b, sizeof(b));
	printf("%c %c", a, b);
	return 0;
}

可不要写成  scanf_s("%c%c",&a,&b,1,1); 这样会出错的

c.代码3

int main()
{
	int a = 0;
	int b = 0;
	scanf_s("%d %d", &a, &b, sizeof(a),sizeof(b));
	printf("%d %d", a, b);
	return 0;
}

 

 注意:如果输入整型数据,scanf函数的里面的sizeof()是在所有取地址的后面,而输入字符,每个取地址后面需要有一个sizeof()

d.代码3

int main()
{
	char arr1[5] = { 0 };
	scanf_s("%s", arr1, sizeof(arr1));
	printf("%s", arr1);
	return 0;
}

注意:输入字符的个数时,一定是比数组的空间个数小的,比如像上面的代码只能输入1-4个字符,而不是5个,不然会出错的。(arr[4]装的是一个'\0',字符串结束标志)

e.代码4

int main()
{
	int a[5] = { 0 };
	for (int i = 0; i < 5; i++)
	{
		scanf_s("%d", &a[i], sizeof(a[i]));
	}
	for (int i = 0; i < 5; i++)
	{
	   printf(" %d", a[i]);
	}
	return 0;
}

注意一点的是 那个取地址符(&)不要忘写了。

f.安全性检测

int main()
{
	char arr1[5] = { 0 };
	int c=scanf_s("%s", arr1, sizeof(arr1));
	printf("%d\n", c);
	printf("%s", arr1);
	return 0;
}

如果输入的字符个数>=数组空间 个数,scanf_s的返回值就为0,反之则为1。

int main()
{
	char arr1[5] = { 0 };
	scanf_s("%s", arr1, sizeof(arr1));
	printf("%s", arr1);
	return 0;
}

 如果输入的字符个数>=数组空间 个数,是不会打印数组中的元素的。(而scanf函数就会打印出)

2.scanf(标准的库函数)

a.代码1

#define _CRT_SECURE_NO_WARNINGS
#include <stdio.h>
int main()
{
	int a = 0;
	scanf("%d", &a);
	printf("%d", a);
	return 0;
}

在VS13 17 19的版本下使用scanf函数一定在最开始(#define _CRT_SECURE_NO_WARNINGS

),不然会报错的。

b.返回值测试

代码1:

int main()
{
	int a = 0;
	int ret=scanf("%d", &a);
	printf("%d\n", ret);
	return 0;
}

 

代码2: 

int main()
{
	int a = 0;
	int b = 0;
	int ret = scanf("%d %d", &a,&b);
	printf("%d\n", ret);
	return 0;
}

代码3: 

int main()
{
	char a = 0;
	int ret=scanf("%c", &a);
	printf("%d", ret);
	return 0;
}

 

代码4:

int main()
{
	char a = 0;
	char b = 0;
	int ret = scanf("%c%c", &a,&b);
	printf("%d", ret);
	return 0;
}
//int main()

 

代码5:

int main()
{
	char str[5] = { 0 };
	int ret=scanf("%s", str);
	printf("%s\n", str);
	printf("%d\n", ret);
	return 0;
}

总结:scanf函数的返回值与函数里面格式符的个数有关 ,返回值即格式符的个数。

c.安全性测试

int main()
{
	char arr1[5] = { 0 };
	int c=scanf("%s", arr1);
	printf("%d\n", c);
	printf("%s", arr1);
	return 0;
}

 当我们输入abcdefg 7个字符时(已经超出了数组的空间个数了),虽然可以打印出来,但是arr1附近的栈空间已被破坏了,出现越界的情况了。scanf_s不会打印出来,也就不存在越界的行为了。

d.当在scanf函数里面输入格式符

如%d %d(%d%d之间有空格),那么通过标准输入设备(键盘)输入时两个整型之间也加一个空格。

如%d,%d(%d%d之间逗号),那么通过标准输入设备(键盘)输入时两个整型之间也加一个逗号。

3.总结

a.使用scanf_s函数虽然安全性比较高,但是可移植性差。比如含有scanf_s函数的代码不一定在其它的编译器上能够跑过。

b.scanf函数虽然没有scanf_s函数那么安全,但可移植性是比scanf_s强。

  scanf_s函数与scanf函数的使用与比较今天就分享到这里了,如果对你有帮助的话吗,可以给个关注,顺便给个赞。

scanfscanf_s都是C语言中用于输入的函数,用来从标准输入流读取数据。二者的使用方式和参数都有所不同。 scanf函数是C语言标准库提供的函数使用较为简单,通常用于读取基本数据类型(如整型、浮点型、字符型等)。它的函数原型为: int scanf(const char *format, ...); 其中,format参数是一个格式字符串,用来指定输入的数据类型和格式,其他参数是读取到的数据的地址。scanf函数会根据format字符串的格式要求,将输入的内容转换为对应的数据类型,并存储到指定的地址中。 而scanf_s函数是安全版本的scanf函数,它在读取字符串时会进行安全性检查,以防止缓冲区溢出的问题。scanf_s的函数原型为: int scanf_s(const char *format, ...); scanf_s函数使用方式scanf函数基本相同,但是在读取字符串时需要指定缓冲区的大小。 具体来说,scanf_s函数需要在读取字符串时指定缓冲区的大小,以避免发生缓冲区溢出的问题。因此,scanf_s函数的格式字符串中需要包含一个指定缓冲区大小的修饰符“%s”之前添加一个整数参数。例如,使用scanf_s函数读取一个字符串的代码可以如下所示: char arr[10]; scanf_s("%9s", arr, sizeof(arr)); 上述代码中,"%9s"表示读取一个最多包含9个字符的字符串,并将其存储到arr数组中。由于arr数组的大小为10,因此可以确保读取的字符串不会导致数组溢出。 综上所述,scanfscanf_s都是用于输入的函数,其中scanf_s是scanf的安全版本,用于防止缓冲区溢出的问题。scanf函数在各种编译器上都能够通用,而scanf_s函数在一些编译器上可能不支持。因此,在编写代码时需要根据具体情况选择适合的函数来进行输入操作。
评论 46
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值