为什么我的变量在strtok()和fgets()之后没有修改就改变了?

时间:2015-06-19 09:28:06

标签: c file text undefined-behavior c-strings

此处演示问题:http://goo.gl/71U1xA

我正在阅读一个文件,在该文件中有一行:

SECTIE FIELD_IN #define ENDSEC

表示我需要将此行后面的行存储到变量中,但如果行包含“#define”部分,则

所以,我正在通过该文件进行迭代,当该行包含“SECTIE”和“FIELD_IN”时,它会检索其中包含#的单词并将其存储在变量keyWord

现在的问题是:keyWord中的值在下一个循环中发生变化而不进行修改。

我的文本文件是:

SECTIE FIELD_IN1 #define ENDSEC
#define COL1                1,1
#define COL2                2,3
#define COL3                5,3
...etc

我的代码输出是:

START

reading line
text=SECTIE FIELD_IN1 #define ENDSEC

keyword1=..
new sectie
keyword2=.#define.

reading line
text=#define COL1              1,1

keyword1=.            1,1
.
start = 1!
keyword3=.            1,1
.

我的代码:

int     x, status, start;
char    inputLine[5000 + 1];
char    copyLine[5000 + 1];
char    *keyWord = "";
FILE    *iFile;

status  = NOERROR;
x       = 0;
start   = 0;

iFile = fopen(gsz_inputFile, "r");  
if(iFile == NULL){ 
    status = ER_OPEN;
    return status;
}

while (fgets(inputLine, sizeof(inputLine), iFile) != NULL){

    printf("\n\nreading line");

    printf("\ntext=%s", inputLine);

    printf("\nkeyword1=.%s.", keyWord);

    strcpy(copyLine, inputLine);
    strlwr(copyLine);
    if(strstr(copyLine, "sectie") != NULL && strstr(copyLine, "field_in") != NULL){
        start = 1;

        printf("\nnew sectie");

        //get keyword
        keyWord = strtok(inputLine," ");
        while (keyWord != NULL)
        {
            if(strstr(keyWord, "#") != NULL){               
                break;
            }

            keyWord = strtok(NULL, " ");
        }

        printf("\nkeyword2=.%s.", keyWord); //here the keyword is correct
        continue;
    }

    if(start){
        printf("\nstart = 1!");

        printf("\nkeyword3=.%s.", keyWord);

        //status = storeString(inputLine, keyw, x); //my actual code is different
        if(status != NOERROR){ x--; }

        x++;
    }       
}

我认为它与while (fgets(inputLine, sizeof(inputLine), iFile) != NULL)有关,因为在执行该行之后,值会发生变化。

为什么keyWord的值在进入下一个循环后会发生变化?我猜它与未定义的行为有关,但我不能指责问题。

2 个答案:

答案 0 :(得分:4)

你有inputLine这是一个数组。在其中,您存储由fgets()提取的字符串:

while (fgets(inputLine, sizeof(inputLine), iFile) != NULL){

在循环中,你说keyWord指向某个inputLine的标记(这意味着它指向数组内的某个位置)。

//get keyword
keyWord = strtok(inputLine," ");

当您运行第二个fgets()数组内容的更改而不是其地址时,keyWord仍指向inputLine内容的已更改

示例:

first fgets():
inputLine: ['H', 'e', 'l', 'l', 'o', ' ', 'W', 'o', 'r', 'l', 'd']
keyWord-->""
second strtok:
keyWord------------------------------------^

second fgets():
inputLine: ['T', 'h', 'i', 's', ' ', 'c', 'h', 'a', 'n', 'g', 'e']
keyword------------------------------------^ (before strtok)

您假设“ keyWord的值已更改”为false:关键字的值仍然相同,您可以使用printf("value of keyWord: %p\n", keyWord);进行检查。更改的是值keyWord 指向,这是您在printf中使用%s显示的内容。

答案 1 :(得分:3)

strtok修改输入缓冲区。因此,在循环结束时得到的keyword是指向inputLine内位置的指针。当循环使用fgets读取另一行时,它会被覆盖。