fscanf_s与scanf_s的宽度参数与缓冲区参数分析

fscanf_s函数 

在文件操作中经常会用到fscanf这个函数,但是在VC和VS中会有警告

意思是编译器觉得fscanf不安全,叫你考虑用一下fscanf_s这个函数来代替fscanf,fscanf_s比fscanf多使用了一个参数

fscanf函数原型:

fscanf(_Inout_ FILE * _File, _In_z_ _Scanf_format_string_ const char * _Format, ...);

fscanf_s函数原型:

fscanf_s(_Inout_ FILE * _File, _In_z_ _Scanf_s_format_string_ const char * _Format, ...);

需要注意的是,使用fscanf时,也可以加第四个参数,编译器是不会报错的(只不过完全没有用),在使用fscanf_s时,第四个参数也可以不写,编译器也不会报错,没有第四个参数的fscanf_s相当于fscanf

意思是,下面这两种写法是完全可以的

第一种:

FILE *fp;
char ss[10];

fp = fopen("E:\\ww.txt", "r+");
fscanf_s(fp , "%s", ss);

 第二种:

fscanf(fp , "%s", ss, 10);//10这个参数完全没有作用

fscanf_s的第四个参数是缓冲区的大小也可以叫最多读取的字符数(注意:这个缓冲区的大小是包括的末尾的('\0')NULL的),下面我们来简单测试一下第四个参数,先上代码:

FILE *fp;
char ss[10];

fp = fopen("E:\\ww.txt", "r+");
fscanf_s(fp , "%s", ss, 10);

printf("%s", ss);

 ww.txt文件中的内容为

运行结果为:

第四个参数10指的是缓冲区的大小,是包含了字符串末尾的'\0'的,所以实际上最多可以读取的字符只有10-1=9个字符,如果读取的字符串长度大于9,那么fscanf_s不会向ss数组中读入任何内容,只

将数组第一个元素赋值为'\0',下面来测试一下

将ww.txt中的内容改为

运行结果:

可以使用断点调试来观察数组ss的的值

还有一点必须要注意,第四个参数,缓冲区的大小必须要小于等于数组的长度,意思是下面这种写法是不允许的

char ss[10];
fscanf_s(fp , "%s", ss, 11);

编译阶段不会报错,但是无法运行(无论读取内容的是否大于缓冲区大小

还有一种方法可以让fscanf_s在读取超过缓冲区大小的内容时,仍然只读取一部分数据,方法如下

fscanf_s(fp , "%5s", ss, 10);

与上面不同的是,我们在格式控制符%s这个加了一个宽度控制,意思是最多读取5个字符

此时ww.txt的内容为明显文本文件的字符数大于了10,但是由于我们加了宽度控制,fscanf_s仍然会读取5个字符到ss数组中

运行效果:

scanf_s函数

 scanf_s的缓冲区大小的控制和fscanf_s完全一样这里就不再赘述了,读者可以看一微软官方文档中对buffer size的描述

重点:一、buffer size包括了末尾的NULL          

   二、buffer size参数描述的允许读取的最大的字符数,而不是字节数

猜你喜欢

转载自www.cnblogs.com/lanhaicode/p/10742790.html