gets()从标准设备读取字符直到遇到换行符(‘\n’)为止;
puts()将buffer中的字符输出到标准输出,直到遇到空字符(’\0’)为止
1. 问:请找出下面代码里的问题:
#include
int main(void)
{
char buff[10];
memset(buff,0,sizeof(buff));
gets(buff);
//思考:不溢出的情况下,strlen(buff)的值是多少?
printf("\n The buffer entered is [%s]\n",buff);
return 0;
}
解析:
gets():
gets是字符串数组操作,使用前必须定义字符串数组。如果输入的字符个数大于定义的数组长度,则会引起危险。
因为gets不会判断缓存是否溢出,会把数组后面的内存覆盖。
//gets 源
void mygets(char * str)
{
int i = 0;
while((*(str + i) = getchar()) != '\n')
{
i++;
}
*(str + i) = '\0';
}
//puts 源
void myputs(char * str)
{
int i = 0;
while( *(str+i) != '\0')
{
putchar(*(str + i));
i++;
}
}
2. scanf()和gets()比较:
scanf虽然也能输入字符串,但是遇到空格就停止了。
两者在接受字符串时:
1.不同点:
scanf不能接受空格、制表符Tab、回车等;
gets能够接受空格、制表符Tab等;
2.相同点:
字符串接受结束后自动加’\0’。
//scanf :当遇到回车,空格和tab键会自动在字符串后面添加’\0’,但是回车,空格和tab键仍会留在输入的缓冲区中
//gets:可接受回车键之前输入的所有字符,并用’\0’替代 ‘\n’.回车键不会留在输入缓冲区中
3. 编写mygets()函数:输入的字符串可以有空格,且,会检测是否溢出,如果溢出,将其截断。也可以设置你想要获取的字符个数,将超出此范围的字符截断。输出的字符串跟输入的对应。
为了达到上述要求,我用getchar函数接收字符,因为getchar是只接收一个字符,所以我用一个while循环检测,让它遇到’\n’或者当字符串接收长度大于规定长度len - 1(因为最后一个存‘\n’),让其停止。
输出函数同理。
缺点:如果len的取值大于数组的长度,仍然导致溢出错误。所以在定义len的长度的时候一定不能大于sizeof(数组名)/sizeof(数组类型)。
优点:如果len的取值合理,既能避免内存溢出问题,又能得到预期效果。
//mygets
void mygets(char *str, int len)
{
int i = 0;
/*if (len < 1)
{
len = 1;
}*/
while ((*(str + i) = getchar()) != '\n' && i < len - 1 )
{
i++;
}
if (i == len - 1) //如果输入的字符串个数大于等于len - 1
{
if ( *(str + i) != '\n') //如果最后一个字符不为'\n',则将其赋为'\n'且清空缓存区
{
while((*(str + i) = getchar()) != '\n');
}
}
}
void myputs(char *str)
{
int i = 0;
while ( putchar(*(str + i++)) != '\n');
}