2

下面是一个程序的函数:

//read the specified file and check for the input ssn
int readfile(FILE *fptr, PERSON **rptr){
    int v=0, i, j;
    char n2[MAXS+1], b[1]=" ";

    for(i=0; i<MAXR; i++){
        j=i;
        if(fscanf(fptr, "%c\n%d\n%19s %19s\n%d\n%19s\n%d\n%19s\n%19s\n%d\n%d\n%19s\n\n",
                    &rptr[j]->gender, &rptr[j]->ssn, rptr[j]->name, n2, &rptr[j]->age, 
                    rptr[j]->job, &rptr[j]->income, rptr[j]->major, rptr[j]->minor,
                    &rptr[j]->height, &rptr[j]->weight, rptr[j]->religion)==EOF) {
            i=MAXR;
        }
        strcat(rptr[j]->name, b);
        //strcat(rptr[j]->name, n2);
        if(&rptr[MAXR]->ssn==&rptr[j]->ssn)
            v=j;
    }
    return v;
}

注释行是这样的,因为由于某种原因,数组“b”包含字符串“n2”,尽管明显缺少赋值。这发生在第一次 strcat 调用之前,但在 fscanf 调用之后/期间。

它确实实现了预期的目标,但是为什么将 n2 连接到 b 的末尾,尤其是当 b 只为 1 个数组元素保留空间时?

以下是 fscanf 调用后的变量定义片段:

*rptr[j]->name = "Rob"

b = " Low"

n2= "Low"
4

2 回答 2

3

它有效,因为你很幸运。 b并且n2碰巧在记忆中彼此相邻,以正确的顺序排列。C 不对数组进行边界检查,并且很乐意让您溢出它们。所以你可以像这样声明一个数组:

char someArray[1] = "lots and lots of characters";

C 编译器(当然是旧的)会认为这很好,即使显然没有足够的空间someArray来存储那么多字符。我不确定它是否定义了它在这种情况下会做什么(我怀疑没有),但在我的编译器上,它将人口限制在数组的大小,所以它不会溢出边界(someArray=={'l'})。

你的情况是一样的(虽然不那么极端)。 char b[1]正在创建一个有足够空间存储 1 个字节的数组。您在该字节中放置了一个空格,因此空终止符没有空间。 strcat, 一直复制内存,直到它到达一个空终止符,因此它会继续运行直到找到一个,即使它直到下一个字符串的结尾(这就是你的情况)。

如果您一直在使用 C++ 编译器,它至少会抛出一个警告(或更可能是一个错误),告诉您您试图将太多项目放入数组中。

于 2011-05-23T14:37:41.253 回答
2

B 需要大小为 2,空格为 1,null 为 1。

于 2011-05-23T14:14:12.533 回答