C 语言 - 结构体与动态内存
结构体与动态内存
您也可以将动态内存与结构体结合使用。
当您事先不知道需要多少个结构体,或者希望通过仅分配必要的内存来节省内存时(例如,在一个汽车经销商程序中,汽车数量不固定),这非常有用。
为结构体分配内存
您可以使用 malloc() 函数为结构体指针分配内存:
实例
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
struct Car {
char brand[50];
int year;
};
int main() {
// 为一个 Car 结构体分配内存
struct Car *ptr = (struct Car*) malloc(sizeof(struct Car));
// 检查分配是否成功
if (ptr == NULL) {
printf("Memory allocation failed.\n");
return 1; // 退出程序并返回错误代码
}
// 设置值
strcpy(ptr->brand, "Honda");
ptr->year = 2022;
// 打印值
printf("Brand: %s\n", ptr->brand);
printf("Year: %d\n", ptr->year);
// 释放内存
free(ptr);
return 0;
}
示例解释
malloc()为一个结构体分配内存strcpy()用于将字符串复制到brand字段中- 我们使用
->通过指针访问成员 - 最后使用
free()释放内存
注意:malloc() 分配未初始化的内存。在您赋值之前,内容将是未定义的。如果您希望内存初始化为零,可以使用 calloc()。
使用结构体数组
您也可以一次为多个结构体分配内存,就像一个数组:
示例:为 3 辆汽车分配内存
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
struct Car {
char brand[50];
int year;
};
int main() {
struct Car *cars = (struct Car*) malloc(3 * sizeof(struct Car));
if (cars == NULL) {
printf("Memory allocation failed.\n");
return 1; // 退出程序并返回错误代码
}
// 填充数据
strcpy(cars[0].brand, "Ford");
cars[0].year = 2015;
strcpy(cars[1].brand, "BMW");
cars[1].year = 2018;
strcpy(cars[2].brand, "Volvo");
cars[2].year = 2023;
// 打印数据
for (int i = 0; i < 3; i++) {
printf("%s - %d\n", cars[i].brand, cars[i].year);
}
free(cars);
return 0;
}
之后使用 realloc() 扩展数组
如果您稍后需要更多元素,可以使用 realloc() 调整动态数组的大小。这可能会将内存块移动到新位置并返回一个新指针。始终先将结果存储在一个临时指针中,以避免在重新分配失败时丢失原始内存。
示例:扩展结构体数组
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
struct Car {
char brand[50];
int year;
};
int main() {
int count = 2;
struct Car *cars = (struct Car*) malloc(count * sizeof(struct Car));
if (cars == NULL) {
printf("Initial allocation failed.\n");
return 1;
}
// 初始化前 2 辆汽车
strcpy(cars[0].brand, "Toyota"); cars[0].year = 2010;
strcpy(cars[1].brand, "Audi"); cars[1].year = 2019;
// 需要多一辆汽车 -> 扩展到 3
int newCount = 3;
struct Car *tmp = (struct Car*) realloc(cars, newCount * sizeof(struct Car));
if (tmp == NULL) {
// 'cars' 在此处仍然有效;释放它以避免内存泄漏
free(cars);
printf("Reallocation failed.\n");
return 1;
}
cars = tmp; // 使用重新分配的内存块
// 初始化索引为 2 的新元素
strcpy(cars[2].brand, "Kia");
cars[2].year = 2022;
// 打印所有汽车
for (int i = 0; i < newCount; i++) {
printf("%s - %d\n", cars[i].brand, cars[i].year);
}
free(cars);
return 0;
}
注意:realloc() 添加的新空间是未初始化的。在使用之前,请确保初始化新添加的元素(例如 strcpy(cars[2].brand, "Kia"); cars[2].year = 2022;)。