2015-04-05 57 views
0

我试图扫描以下格式在C一个文本文件:功能的fscanf在.txt文件中忽略空格分隔符

key01 value01 
key02 value02 
key03 value03 

我想利用关键字和值字符串,并将其分配给一个struct我创建:

struct Mapping 
{ 
    char key[sizeof(char) * 5]; 
    char value[sizeof(char) * 7]; 
} MapElement; 

我的代码要做到这一点是:

void initialize_server_map(struct Mapping serverMap[]) 
{ 
    char* fileName = "server1.txt"; 
    FILE *fp = fopen(fileName, "r"); 
    char *lineKey = (char*)malloc(sizeof(char) * 5); 
    char *lineValue = (char*)malloc(sizeof(char) * 7); 
    int i = 0; 
    while (fscanf(fp, "%s %s", lineKey, lineValue) != EOF) { 
    struct Mapping mapping; 
    memcpy(mapping.key, lineKey, sizeof(char) * 5); 
    memcpy(mapping.value, lineValue, sizeof(char) * 7); 
    printf("key: %s value: %s\n", mapping.key, mapping.value); 
    serverMap[i] = mapping; 
    i++; 
    } 
    fclose(fp); 
} 

int main(void) 
{ 
    int size = 4; 
    struct Mapping serverMap[size]; 
    initialize_server_map(serverMap); 
} 

我的输出如下:

key: key01value01 value: value01 
key: key02value02 value: value02 
key: key03value03 value: value03 
key: key04value04 value: value04 

当然,我想这个输出,而不是:

key: key01 value: value01 
key: key02 value: value02 
key: key03 value: value03 
key: key04 value: value04 

我敢肯定它做我分配我的记忆方式的东西。有人可以帮我解决这个问题吗?另外,有没有更有效的方法来做到这一点?

+0

另外,我知道一个事实,即它们键和值将始终是长度段5和7 – 2015-04-05 05:01:00

+0

在C语言中,字符串终止NUL字节,占用一个字节。您应该(几乎)从不使用'memcpy'来复制字符串;使用'strcpy'这将纠正NUL终止副本。请记住为阵列和malloc中的NUL字节留出空间。而'char [sizeof(char)* 5]'显然是错误的。声明中的数字是单位的数量,而不是字节的数量(所以'int v [5]'有足够的空间存放5个整数,不管整数可能有多大)。幸运的是,sizeof(char)总是一个。 – rici 2015-04-05 05:03:28

+0

如果您打算将它们视为字符串,则忘记为必需的'null-terminating'字符添加'+ 1'。 – 2015-04-05 05:03:39

回答

1

由于无法访问数组,导致未定义的行为。

为了存储"key01",您需要一个至少为6 - 5的字符数组和一个用于终止空字符的数组。

同样,您需要一个大小至少为8的数组来存储"value01"

您正在声明大小分别为57。因此,你在修改数组越界时,在使用

while (fscanf(fp, "%s %s", lineKey, lineValue) != EOF) { 
1

那么,至少你已经忘记了字符串尾随'\ 0'的空间。所以如果你有5个字符的字符串,你应该分配6个字符。

但通常整个方法都很脆弱。如果输入数据中的键长度超过5个字符会怎么样?传统的方法是使用“足够大”的缓冲区(例如几百字节),使用格式中的长度限制器(例如“%20s”而不是“%s”)限制可能的字符串长度,然后分配需要的空间通过strdup

所以整个大块看起来像:

struct Mapping { 
    char* key; 
    char* value; 
}; 
... 
char lineKey[51]; 
char lineValue[51]; 
while (fscanf(fp, "%50s %50s", lineKey, lineValue) != EOF) { 
    struct Mapping* mp = (struct Mapping*)malloc(sizeof(struct Mapping)); 
    assert(mp); 
    mp->key = strdup(lineKey); 
    mp->value = strdup(lineValue); 
    ... 
}