fgets和gets的区别.docx
- 文档编号:15651244
- 上传时间:2023-07-06
- 格式:DOCX
- 页数:21
- 大小:41.95KB
fgets和gets的区别.docx
《fgets和gets的区别.docx》由会员分享,可在线阅读,更多相关《fgets和gets的区别.docx(21页珍藏版)》请在冰点文库上搜索。
fgets和gets的区别
fgets函数
从流中读一行或指定个字符,
原型是char*fgets(char*s,intn,FILE*stream);
从流中读取n-1个字符,除非读完一行,参数s是来接收字符串,如果成功则返回s的指针,否则返回NULL。
形参注释:
*string结果数据的首地址;n-1:
一次读入数据块的长度,其默认值为1k,即1024;stream文件指针
例:
如果一个文件的当前位置的文本如下
Love,IHave
But........
如果用
fgets(str1,4,file1);
则执行后str1="Lov",读取了4-1=3个字符,
而如果用
fgets(str1,23,file1);
则执行str="Love,IHave",读取了一行(不包括行尾的'n')。
序例:
#include
#include
intmain(void)
{
FILE*stream;
charstring[]="Thisisatest";
charmsg[20];
/*openafileforupdate*/
stream=fopen("DUMMY.FIL","w+");
/*writeastringintothefile*/
fwrite(string,strlen(string),1,stream);
/*seektothestartofthefile*/
fseek(stream,0,SEEK_SET);
/*readastringfromthefile*/
fgets(msg,strlen(string)+1,stream);
/*displaythestring*/
printf("%s",msg);
fclose(stream);
return0;
}
fgets函数用来从文件中读入字符串。
fgets函数的调用形式如下:
fgets(str,n,fp);此处,fp是文件指针;str是存放在字符串的起始地址;n是一个int类型变量。
函数的功能是从fp所指文件中读入n-1个字符放入str为起始地址的空间内;如果在未读满n-1个字符之时,已读到一个换行符或一个EOF(文件结束标志),则结束本次读操作,读入的字符串中最后包含读到的换行符。
因此,确切地说,调用fgets函数时,最多只能读入n-1个字符。
读入结束后,系统将自动在最后加'\0',并以str作为函数值返回。
------------------------------------------------------------------------------------
在C语言,常用的同类函数还有fscan()和gets(),不过这两种函数都有其缺点,或者可以说是制约性。
fscan()函数不能读取带有空格的字符串,当字符串中含有空格时,函数只能读取空格之前的字符串,空格之后的全部忽略。
gets()函数虽然可以操做带有空格的字符串,不过容易引起溢出,因为C语言是将字符串当做一种数组操做的,既然是数组,就有长度,如果gets()函数读取的字符串长度大于被赋值数组的长度,就会引起溢出,而且新版的GCC已经不能正确编译gets()函数了,会报错。
fgets()函数的基本用法为:
fgets(char*s,intsize,FILE*stream);
括号中,第一个参数为将被赋值的数组名,这里需要注意的是,其中不用写数组名称后面的方括号以及其中的数组长度。
第二个参数是将要读取字符串的长度,这里需要注意的是,这里长度的数值是“字符串实际长度+1”,加1是因为,字符串最后面还有一个/0位。
最后一个参数为输入设备或者变量。
一般情况下,我们都是从键盘输入,也就是stdin。
(从文件读取的这里暂不说明)
OK!
现在举一个例子!
下面的这段代码作用在于,属于一个字符串,然后计算出字
符串中单词的个数。
#include
main()
{
chardia[80];
chart,m;
inti=0,num=0,flag=0;
fgets(dia,80,stdin);
fputs(dia,stdout);
for(i=0;dia[i]!
=’\0′;i++)
{
if(dia[i]==’‘)
flag=0;
else
if(flag==0)
{
flag=1;
num++;
}
}
printf("%d\n",num);
}
这段代码的工作原理是,使用一个用于标示的变量flag,还有一个用于统计单词个数的变量num。
然后逐一读取字符串中的每一个字符,当遇到字母时flag的值为1,当遇到空格时,flag的值为0。
如果遇到一个字母,并且此时flag值为0(说明这个字母之前的一个字符是空格),给num加1,直到遇到空格时,将flag重新赋值为0。
在fgets()使用中需要注意,其中第二个参数规定的字符串长度应该与字符串长度相等,否则运行程序时会有溢出的错误。
当输入的字符串大于限定的字符串长度时(本例中限定的是80),限定长度之后的字符串会被丢弃。
fgets()函数遇到换行终止~
------------------------------------------------------------------------------------------------
区分C语言中getch、getche、fgetc、getc、getchar、fgets、gets
首先,这两个函数不是C标准库中的函数,
intgetch(void) //从标准输入读入一个字符,当你用键盘输入的时候,屏幕不显示你所输入的字符。
也就是,不带回显。
intgetche(void) //从标准输入读入一个字符,键盘输入的时候,屏幕显示所输入的字符。
带回显。
这两个函数包含在头文件conio.h中,需要记住的是conio.h不是C标准库中的头文件。
Micorsoft和Borland的C编译器提供了conio.h,用来创建控制台文本用户界面。
一般在Windows系统下安装了VS、VC等,就可以包含conio.h头文件。
但是一般在Unix、Linux系统中,/usr/include/中都没有这个头文件。
getch和getche在等待用户从键盘输入的时候,用户按下一个键后,不需要按回车,程序自动往下执行。
在Linux中,终端输入在缺省情况下是被“一锅端”的,也就是说整行输入是被一起处理的。
通常,这是一种人们所希望的方便的办法,但它也意味着在读入数据时必须按一下回车键表示输入行结束后才能得到输入的数据。
在游戏中,很多都提供了“老板键”,它的实现,就是利用了这两个函数。
其次,除了getch和getche,其他的都是C标准库中的头文件,包含在头文件stdio.h中。
intfgetc(FILE*stream);//从流stream中读一个字符。
可以将标准输入stdin作为它的实参,这时候从标准输入读取一个字符。
intgetc(FILE*stream);//和fgetc等效,由fgetc通过宏实现。
intgetchar(void);
//从标准输入stdin读入一个字符,程序等待你输入的时候,你可以输入多个字符,回车后程序继续执行。
//但getchar只读入一个字符说明:
getc、getchar都是通过宏定义借助fgetc实现。
如getchar的实现为,#definegetchar()fgetc(stdin)。
char*fgets(char*str,intnum,FILE*stream); //从流stream中读入最多num个字符到字符数组str中,当遇到换行符时、或读到num-1个字符时停止。
//自动加上'\0'空字符结尾
char*gets(char*str); //从标准输入stdin读取一个字符串,遇到换行或结束时候终止。
//不同于fgets,他没有指定num,所以需要注意字符数组str的大小。
说明:
fgets和gets之间没有宏定义的关系,彼此各自有自己的实现。
蠕虫病毒的实现就是函数gets的“功劳”。
gets函数的任务是从流中读入一个字符串。
它的调用者会告诉它把读入的字符串放在什么地方。
但是,gets()函数并不检查缓冲区大小,如果调用者提供了一个指向堆栈的指针,并且get()函数读入的字符数量超过了超过了缓冲区的空间大小,get()会愉快地将多出来的字符继续写入到堆栈中,这就覆盖了堆栈中原来的内容。
如:
main(){charline[512];//在程序的堆栈上分配512个字符的空间...gets(line);//蠕虫病毒的入口,可以将恶意代码通过多出来的数据写入堆栈}
建议不要用getch和getche,因为它们不是C标准库中的函数。
用它们写出的程序可移植性差,不同的编译器不保证可以包含conio.h。
建议用fgets函数彻底替代gets函数。
另外,绝大多数的这些get函数,都有对应的put版本。
intfputc(intcharacter,FILE*stream);
intputc(intcharacter,FILE*stream);//通过宏定义和fputc实现
intputchar(intcharacter); //通过宏定义实现:
#defineputchar(c)fputc(c,stdout)
intfputs(constchar*str,FILE*stream);
intputs(constchar*str);
说明:
两者之间无宏定义实现关系。
puts(constchar*str)近似等效于fputs(cosntchar*str,stdout),不同点是前者还输出一个'\n'
最后,关于EOF
EOF是在stdio.h文件中定义的符号常量,值为-1。
如,fputc函数返回一个值:
如果输出成功则返回值就是输出的字符;如果输出失败,则返回一个EOF。
fgetc函数读字符时遇到文件结束符,函数返回一个文件结束标记EOF。
如果想从一个磁盘文件顺序读入字符并在屏幕上显示,可以:
ch=fgetc(fp);
while(ch!
=EOF){
putchar(ch);
ch=fgetc(fp);
}
注意,EOF不是可输出字符,因此不能在屏幕上显示。
由于ASCII码不可能出现-1,因此EOF定义为-1是合适的。
当读入的字符值等于-1(即EOF)时,表示读入的已不是正常的字符,而是文件结束符。
但以上只适用于读取文本文件的情况。
现在ANSIC已经允许用缓冲文件系统处理二进制文件,而读入某一个字节中的二进制数据的值有可能是-1,而这又恰好是EOF的值。
这就出现了需要读入有用数据,却处理为“文件结束”。
feof(fp)用来测试fp所指向的文件当前状态是否是“文件结束”。
如果想顺序读入一个二进制文件数据,可以:
while(!
feof(fp)){
c=fgetc(fp);
...
}
------------------------------------------------------------------------------------------------
函数名:
fgets
功能:
从流中读取一字符串
用法:
char*fgets(char*string,intn,FILE*stream);
形参注释:
*string结果数据的首地址;n-1:
一次读入数据块的长度,其默认值为1k,即1024;stream文件指针
序例:
#include
#include
intmain(void)
{
FILE*stream;
charstring[]="Thisisatest";
charmsg[20];
/*openafileforupdate*/
stream=fopen("DUMMY.FIL","w+");
/*writeastringintothefile*/
fwrite(string,strlen(string),1,stream);
/*seektothestartofthefile*/
fseek(stream,0,SEEK_SET);
/*readastringfromthefile*/
fgets(msg,strlen(string)+1,stream);
/*displaythestring*/
printf("%s",msg);
fclose(stream);
return0;
}
fgets函数fgets函数用来从文件中读入字符串。
fgets函数的调用形式如下:
fgets(str,n,fp);此处,fp是文件指针;str是存放在字符串的起始地址;n是一个int类型变量。
函数的功能是从fp所指文件中读入n-1个字符放入str为起始地址的空间内;如果在未读满n-1个字符之时,已读到一个换行符或一个EOF(文件结束标志),则结束本次读操作,读入的字符串中最后包含读到的换行符。
因此,确切地说,调用fgets函数时,最多只能读入n-1个字符。
读入结束后,系统将自动在最后加'\0',并以str作为函数值返回。
编辑本段]
fgets(由文件中读取一字符串)
相关函数
open,fread,fscanf,getc
表头文件
include
定义函数
har*fgets(char*s,intsize,FILE*stream);
函数说明
fgets()用来从参数stream所指的文件内读入字符并存到参数s所指的内存空间,直到出现换行字符、读到文件尾或是已读了size-1个字符为止,最后会加上NULL作为字符串结束。
返回值
gets()若成功则返回s指针,返回NULL则表示有错误发生。
范例
#include
main()
{
chars[80];
fputs(fgets(s,80,stdin),stdout);
}
执行
thisisatest/*输入*/
thisisatest/*输出*/
------------------------------------------------------------------------------------------------字符和字符串(字符数组)处理
2009-09-1615:
48
几个常见特殊字符的整型数字
描述charint
空格''32
TAB键'\t'9
回车换行LF10
文件结束EOF-1
字符串结束符号'\0'40
TAB字符处理要小心,经过到记事本copy/paste后,TAB键被转化成几个空格
for(;str[i]==''||str[i]=='';i++);
但经过到记事本copy/paste后,TAB键被转化成几个空格
所以系统总报warning:
tmp.c:
58:
warning:
comparisonisalwaysfalseduetolimitedrangeofdatatype
tmp.c:
59:
27:
warning:
characterconstanttoolongforitstype
‘a’与”a”的区别(实际就是字符和字符串的区别)
‘a’
1字节‘a’
”a”
2字节
‘a’’\0’
字符串数组charstr[20],至少有一个str[n]是0
strcpy(str,”123”);
str[0]=’1’
str[1]=’2’
str[2]=’3’
str[3]=0
字符串指针=字符串数组名=字符串数组第一个元素的地址(&)
char*s;
charstr[20];
s=str=&str[0]
字符串数组和字符串指针的相同和不同
作为函数参数,字符串数组名和字符串指针没区别
showtable(chars[50]);
showtable(char*s);
在printf上,字符串数组和字符串指针没区别
chars[20];
char*p;
p="apoint";
sprintf(s,"achararrage");
printf("p'svalueis:
%s\n",p);
printf("s'svalueis:
%s\n",s);
p'svalueis:
apoint
s'svalueis:
achararrage
字符串数组和字符串指针在赋值上不同
字符串数组无法直接赋值,只能用sprintf,strcpy,strcatchars[20];
char*p;
chars[20];
p="apoint";
sprintf(s,"achararrage");
strcpy(s,"achararrage");
字符串指针只支持printf,不支持scanf,scanf还是建议用字符串数组
对指针初始赋值,不分配空间,可以支持printf
char*p;
p="ppp";
printf("p'svalueis:
%s\n",p);
[macg@localhostmysqltmp]$./tt
p'svalueis:
ppp
对指针初始赋值,不分配空间,不支持scanf
char*p;
p="ppp";
scanf("%s",p);
[macg@localhostmysqltmp]$./tt
Segmentationfault
如何使字符串指针支持scanf操作,必须用malloc分配字节
#include
main()
{
chars[20];
char*p;
intret;
p=malloc(20);
scanf("%s",p);
printf("p'svalueis:
%s\n",p);
free(p);
[macg@localhostmysqltmp]$./tt
dddd
p'svalueis:
dddd
对malloc要求:
#include
malloc返回指针p=malloc(20);
记着要free(指针)free(p);
虽然也可以用字符串数组初始化字符串指针,但不建议
实际就等于操作的是字符串数组了,多此一举
chars[20];
char*p;
p=s;
字符串不建议直接赋值和比较,用函数strcpy
strcpy(char*,char*)
字符串拷贝函数
后者拷贝到前者
strcat(char*,char*)
字符串追加函数
后者追加到前者后
strcmp(char*,char*)
对字符串是不允许做==或!
=的运算,只能用字符串比较函数,字符串比较只能采用strcmp
不能用同一个字符串数组赋值给多个字符串指针,这样会造成两个字符串指向同一空间.
char*name,*media,a[30],s[80],c;
name=media=a;
以后修改meida,name也会跟着改变
gets读字符串也有”连读”问题,不过这个影响的不是scanf前面剩下的'\n',而是前面剩下的0
Chara[20];
gets(a)
while(a[0]==0)gets(a);
scanf读字符串,只能读到空格,所以scanf只能读单个单词的字符串,不能读"句子"
chars[80];
scanf("%s",s);
printf("yourinputis:
%s",s);
[macg@localhostmysqltmp]$./tt
abcdefhig
yourinputis:
abc
gets能克服scanf的缺陷,读字符串包含空格
chara[30];
gets(a);
printf("yourinputis:
%s\n",a);
[macg@localhostmysqltmp]$./tt
abcefdsdd
yourinputis:
abcefdsdd
fgets字符串指针改成字符串数组,消灭了Segmentationfault错误
char*re,*rec;
re=fgets(rec,100,srcstream);
出Segmentationfault错误
改成
char*re,rec[100];
re=fgets(rec,100,srcstream);
Poweredby1.5?
2001-2009ComsenzInc.
------------------------------------------------------------------------------------------------
fgets使用注意事项
fgets(buf,size,fp)其行为方式如下:
(1)遇到换行或文件结束EOF则返回。
(2)按行读取。
(3)每一行结束处的换行字符‘\n’也算该行字符。
(4)对于大小为size的buf,最多只读取size-1个字符。
(5)自动地把buf中最后一个字符(通常是换行符)的后面一个字节填充零结束符('\0')。
因此如果想把buf中的换行符去掉,可以这样:
if(buf[strlen(buf)-1]=='\n')
buf[strlen(buf)-1]='\0';
当然前提是buf足够大,可以容纳完整的一行(包括最后的那个换行符)。
fputs(str,fp)
(1)把str中零结束符之前的全部文字输入到文件中。
(2)输入完成后,不会增加额外的特殊字符,如换行符等。
如果想输入完str后,就换行,则应该在上述调用后,再调用fputc('\n',fp);才行。
------------------------------------------------------------------------------------------------
fgets函数与fputs函数
作者:
陈刚,华清远见嵌入式培训中心讲师。
#include
char*fgets(char*s,int
- 配套讲稿:
如PPT文件的首页显示word图标,表示该PPT已包含配套word讲稿。双击word图标可打开word文档。
- 特殊限制:
部分文档作品中含有的国旗、国徽等图片,仅作为作品整体效果示例展示,禁止商用。设计者仅对作品中独创性部分享有著作权。
- 关 键 词:
- fgets gets 区别