作者:赖皮小王子 | 来源:互联网 | 2023-10-13 12:45
gcc版本7.3.0,这个问题困扰了我一整天也没找到原因,问题描述请看照片,i的值为-1,*n的值为10,为什么我输出i>(*n-2)的值为真?我一个个实验,最后发现-1>*n
gcc版本7.3.0,这个问题困扰了我一整天也没找到原因,问题描述请看照片,i的值为-1,*n的值为10,为什么我输出i>(*n-2)的值为真?
我一个个实验,最后发现-1>*n的值也为真,但是如果用正数和*n比较的话结果就正确,比如1>*n的值却为假。
如果我把17行的代码放到当前文件的36行下就不会出这样的问题。请问这是不是编译器出问题了?但是我在mac上编译也是同样的问题。
具体代码如下:
#include
#include
#include
#include
ssize_t mygetline(char **lineptr, size_t *n, FILE *stream)
{
char *tmp=NULL;
char c;
int i;
if(*n==0)
*n=10;
if(*lineptr==NULL)
*lineptr=malloc(*n);
for(i=-1;;)
{
if(i>((*n)-2))
{
*n+=10;
tmp=realloc(*lineptr,*n);
if(tmp==NULL)
return -2;
*lineptr=tmp;
}
c=fgetc(stream);
if(c==EOF)
break;
i++;
if(c=='\n')
{
break;
}
*(*lineptr+i)=c;
}
if(i==-1)
*(*lineptr+i+1)='\0';
else
*(*lineptr+i)='\0';
return i;
}
int main(int argc, char *argv[])
{
char *line = NULL;
size_t linesize = 0;
FILE *fp;
if(argc < 2)
{
fprintf(stderr,"Usage...\n");
exit(1);
}
fp = fopen(argv[1],"r");
if(fp == NULL)
{
perror("fopen()");
exit(1);
}
while(1)
{
if(mygetline(&line,&linesize,fp) < 0)
break;
printf("linesize = %ld\n",linesize);
printf("%ld\n",strlen(line));
printf("%s\n",line);
}
free(line);
fclose(fp);
exit(0);
}
3 个解决方案
你的n是无符号类型,i是有符号类型 ,比较时i会转换成size_t(无符号),对应的并不是负1,而是一个很大的数
编译器不会出问题 的。只是它是按照它的规则去解析和生成可执行程序。
-1 > 8这个表达式是0(false);因为-1和8编译器是按照有符号数比较处理的。同理: -1 > (10 - 8);
但是-1 > (*n - 2)就不一样了。由于*n是size_t类型,size_t是无符号数,对于无符号数比较,编译器会现将他们转换成统一类型,一般都是将类型向高转换(也叫隐式提升);所以,会将-1先转无符号,由于-1是0xffffffff;即将最高位由符号位编程有效数值位,因此得到的是一个很大值。所以-1 > (*n - 2)表达式的值是1(true);
不要迷信书、考题、老师、回帖;
要迷信CPU、编译器、调试器、运行结果。
并请结合“盲人摸太阳”和“驾船出海时一定只带一个指南针。”加以理解。
任何理论、权威、传说、真理、标准、解释、想象、知识……都比不上摆在眼前的事实!