gpt4 book ai didi

c - 在 C 中释放动态分配的结构实例

转载 作者:太空宇宙 更新时间:2023-11-04 03:37:11 25 4
gpt4 key购买 nike

我有以下结构:

typedef struct generic_attribute_struct{
attribute_value current_value;
attribute_value previous_value;
attribute_value running_value;
} generic_attribute;

其中 attribute_value 只是 unsigned long long int 的占位符。

我有这个结构的以下构造函数:

generic_attribute* construct_generic_attribute(attribute_value current_value){
generic_attribute *ga_ptr; //allocate space for pointer to generic attribute
if(ga_ptr = malloc (sizeof (generic_attribute))){ //if allocation succeeds
set_ga_current_value(ga_ptr, current_value); //assigned the current value to given input
set_ga_previous_value(ga_ptr, 0); //set previous value to zero
set_ga_running_value(ga_ptr);
} else{ //else, inform user of error
fprintf(stderr, "Health Monitor ran out of space when allocating memory for a generic attribute.");
}
return ga_ptr; // return pointer to new attribute or a NULL pointer if memory allocation failed
}

set_ga_running_value 看起来像这样:

attribute_value set_ga_running_value(generic_attribute* ga_ptr){
attribute_value delta = get_ga_current_value(ga_ptr) - get_ga_previous_value(ga_ptr);
ga_ptr->running_value = (ga_ptr->running_value? ga_ptr->running_value : 0) + delta;
return ga_ptr->running_value;
}

此结构的析构函数如下所示:

void destroy_generic_attribute(generic_attribute** ga_ptr){
free(*ga_ptr);
}

我创建了一个测试,要求用户输入一些 current_value,构造一个指针,并打印出结构变量的值是否是它们应该的值。这个测试循环直到用户不再想要测试,在这种情况下他们退出测试。

所以,测试看起来像这样:

  1. 用户想测试吗?是,执行2)。否,转7)。
  2. 获取用户的输入
  3. 使用这个新输入调用通用属性的构造函数
  4. 验证是否正确创建了通用属性
  5. 调用泛型属性的析构函数
  6. 转到 1。
  7. 退出

这是测试的样子:

void test_generic_attribute_constructor_with_user_input(){
attribute_value input;
int continue_var;
bool current_value_test, previous_value_test, running_value_test, pointer_test;
generic_attribute* ga_ptr;
while(1){
printf("Would you like to execute a test for Generic Attribute Constructor? Type 1 for YES, or 0 for NO: ");
scanf("%i", &continue_var);
if(continue_var){
printf(TESTING, "Constructor for Generic Attribute and Generic Attribute Reader");
printf("\n" INPUT, "single number");
scanf("%lld", &input);
ga_ptr = construct_generic_attribute(input);
read_generic_attribute(ga_ptr);
current_value_test = (get_ga_current_value(ga_ptr) == input ? true : false);
previous_value_test = (get_ga_previous_value(ga_ptr) == 0 ? true: false);
// THIS TEST FAILS
running_value_test = (get_ga_running_value(ga_ptr) == input ? true: false);
pointer_test = (ga_ptr ? true: false);
printf("%s.\n", ((current_value_test && previous_value_test && running_value_test && pointer_test) ? PASS : FAIL));
destroy_generic_attribute(&ga_ptr);

}else{
printf("\nOK! Testing concluded.");
break;
}

}
}

我的问题是,当 ga_ptr 被销毁时,“运行值”似乎永远不会“重置”。它似乎保留了它的旧值。如何正确清除整个 ga_ptr 结构的内存?

测试结果:

Would you like to execute a test for Generic Attribute Constructor? Type 1 for YES, or 0 for NO: 1
Testing Constructor for Generic Attribute and Generic Attribute Reader:
Please enter single number for input: 10
Generic Attribute has the following contents:

Pointer Current Value Previos Value Running Value
0x600058530 10 0 10
PASS.
Would you like to execute a test for Generic Attribute Constructor? Type 1 for YES, or 0 for NO: 1
Testing Constructor for Generic Attribute and Generic Attribute Reader:
Please enter single number for input: 20
Generic Attribute has the following contents:

Pointer Current Value Previos Value Running Value
0x600058530 20 0 30
FAIL.

我希望运行值是 20。

如果我将析构函数更改为:

void destroy_generic_attribute(generic_attribute** ga_ptr){
set_ga_current_value(*ga_ptr, 0);
set_ga_previous_value(*ga_ptr, 0);
(*ga_ptr)->running_value = 0;
free(*ga_ptr);
}

我的测试通过了...但是,我不明白为什么跳过 setter 会使代码失败。

最佳答案

您只是通过使用从未初始化的值来调用未定义的行为。

construct_generic_attribute 中,您初始化当前值和先前值,然后调用 set_ga_running_value。在后者中,您使用刚刚初始化的当前值和先前值来计算 delta :直到这里都很好。但是你有:

ga_ptr->running_value = (ga_ptr->running_value? ga_ptr->running_value : 0) + delta;

这意味着您在初始化之前使用 running_value。因为它在新 malloc 的结构中,所以它的值只是 undefined。它可能是 0,或者它可能是分配之前存在于该内存位置的值,或者它可能是编译器用作特殊初始化的特殊值:你无法知道也不应该依赖任何东西。

您的编译器似乎在运行时将内存预初始化为 0,然后从不更改 free 和 malloc 的值,在第二次运行时给出 30。我的(在 Debug模式下)总是将 malloc 的值初始化为 0xcdcdcdcd,每次测试都失败。只是未定义的行为......

所以你真的应该在你的构造函数中初始化新分配的结构:

if(ga_ptr = malloc (sizeof (generic_attribute))){       //if allocation succeeds
memset(ga_ptr, 0, sizeof(generic_attribute)); // ensure all values are set to 0
set_ga_current_value(ga_ptr, current_value); //assigned the current value to given input
set_ga_previous_value(ga_ptr, 0); //set previous value to zero
set_ga_running_value(ga_ptr);

或者如果您知道 running_value 是一个整数,只需将 memset 替换为:

ga_ptr->running_value = 0;

关于c - 在 C 中释放动态分配的结构实例,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/31696210/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com