-1

我用 typedef person_t 在一个人结构中创建一个人:

int main(int argc, char* argv[]) {
    person_t a;
    memset(&a, 0, sizeof(person_t));

    person_set_name(&a, "Konrad Hoppenstauffer");
    person_set_age(&a, 42);

void person_set_name(person_t* person, char* name) {
    if(person->name) {
        free(person->name);
    }
    person->name = malloc(sizeof(char) * strlen(name) + 1);
    strcpy(person->name, name);
}

以上工作正常。

使用此功能时会出现问题:

person_t* string_to_person(char* str) { 
    person_t* person = malloc(sizeof(person_t));

    int len = 0;
    while(str[len] != '\t') {
        len++;
    }

    char* name = malloc(len + 1);

    int i;
    for(i = 0; i < len; i++) {
        name[i] = str[i];
    }
    name[len] = '\0';

    person_set_name(person, name);
    person_set_age(person, atoi(str+len+1));

    return person;
}

这里的 str 类似于:“Name Nameson\t22”。那是由制表符分隔的名称。然后我将两者分开并将字符放入 char* 名称中。

person_t 是结构的 typedef。

如果我从 person_set_name 中删除 free(person->name),一切正常。但是如果我把它留在里面,名字就会变成垃圾,例如:“É8>”。

我假设在我复制每个字符的 for 循环中发生了错误。但是以我有限的 CI 经验看不出是什么。帮助表示赞赏。

4

1 回答 1

0

您正在尝试释放垃圾指针。

后:

person_t* person = malloc(sizeof(person_t));

malloc不会使用任何特定数据初始化新内存块,因此您的程序*person此时必须将其视为包含垃圾(因为它可能包含任何数据)。特别是,person->name(ie (*person).name) 可能不是NULL

不久之后,此代码运行:

if(person->name) {
    free(person->name);
}

- 如果person->name不是NULL,那么你释放它。由于person->name没有指向您分配的malloc内容,因此此时您已经完全处于 Undefined Behavior Land™ 中。

一种可能的解决方法是person->name = NULL;在分配人员后立即设置。

于 2015-09-24T02:03:15.487 回答