C语言,怎么为动态结构体数组分配内存

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了C语言,怎么为动态结构体数组分配内存相关的知识,希望对你有一定的参考价值。

#include <stdio.h>
#include <malloc.h>
#include <stdlib.h>
struct record

float coef;
int expn;
;

void main()

int *array = 0, num, i;
printf("please input the number of element: ");
scanf("%d", &num);

/*申请动态数组使用的内存块*/
array = (struct record *)malloc(sizeof(struct record )*num);
if (array == 0) /*内存申请失败,提示退出*/

printf("out of memory,press any key to quit...\n");
exit(0); /*终止程序运行,返回操作系统*/


/*提示输入num个数据*/
printf("please input %d elements: ", num);
for (i = 0; i < num; i++)

scanf("%c", &array[i].coef);
scanf("%d", &array[i].expn);


/*输出刚输入的num个数据*/
printf("%d elements are: \n", num);
for (i = 0; i < num; i++)

printf("%f,", array[i].coef);
printf("%d,", array[i].expn);

printf("\b \n"); /*删除最后一个数字后的分隔符逗号*/
free(array); /*释放由malloc函数申请的内存块*/
getch();


高手帮帮忙 !!怎么修改正确

这行代码:array = (struct record *)malloc(sizeof(struct record )*num);
是要申请struct record类型的内存,而你定义成int*,所以int* array改成struct record* array,另外num和i的定义不要record定义在一起,否则类型和record一样,要独立定义为:int num,i,最后,scanf("%c", &array[i].coef);改成scanf("%f", &array[i].coef);
参考技术A 这行代码:array = (struct record *)malloc(sizeof(struct record )*num);
是要申请struct record类型的内存,而你定义成int*,所以int* array改成struct record* array,另外num和i的定义不要record定义在一起,否则类型和record一样,要独立定义为:int num,i,最后,scanf("%c", &array[i].coef);改成scanf("%f", &array[i].coef);
参考技术B 看看你的部分程序!
struct record

float coef;
int expn;
;
/*提示输入num个数据*/
printf("please input %d elements: ", num);
for (i = 0; i < num; i++)

scanf("%c", &array[i].coef);
scanf("%d", &array[i].expn);

你的结构体中的元素coef是定义成浮点型的,而你在下面输入的时候用的是%c字符方式接收的,这是怎么回事啊!
编程的时候一定要细心,一个标点都不能出错!

C 语言结构体 ( 结构体中嵌套一级指针 | 分配内存时先 为结构体分配内存 然后再为指针分配内存 | 释放内存时先释放 指针成员内存 然后再释放结构头内存 )





一、结构体中嵌套一级指针




1、声明 结构体类型


声明 结构体类型 : 这里注意 , 在结构体中 , 定义一个 一级指针 变量 , 注意与 数组类型区别 ;

  • 结构体内定义数组 , 声明变量时 , 会自动分配数组内存 ;
  • 结构体内定义指针 , 声明变量时 , 只会为 4 字节指针分配内存 ;
/**
 * @brief The Student struct
 * 定义 结构体 数据类型 , 同时为该结构体类型声明 别名
 * 可以直接使用 别名 结构体变量名 声明结构体类型变量
 * 不需要在前面添加 struct 关键字
 */
typedef struct Student

    // 声明变量时 , 会自动分配这 5 字节内存
    // 赋值时 , 可以直接使用 = 赋值字符串
    char name[5];
    int age;
    int id;
    // 声明变量时 , 只会为 4 字节指针分配内存
    // 具体的 字符串内存 需要额外使用 malloc 申请内存
    // 赋值时 , 必须使用 strcpy 函数 , 向堆内存赋值
    char *address;
Student;

2、为 结构体 变量分配内存 ( 分配内存时先 为结构体分配内存 然后再为指针分配内存 )


为 结构体 变量分配内存 : 结构体 内存分配完成之后 , 需要立刻为 结构体的 一级指针 成员分配内存 ;

/**
 * @brief create_student 堆内存中分配内存
 * @param array 二级指针 , 指向结构体数组
 * @return
 */
int create_student(Student **array, int count)

    // 返回值
    int ret = 0;
    // 循环控制变量

    int i = 0;
    // 临时变量
    Student *tmp = NULL;

    // 验证二级指针合法性
    if(array == NULL)
    
        ret = -1;
        return ret;
    

    // 堆内存中申请内存
    tmp = (Student *)malloc(sizeof(Student) * count);

    // 初始化分配的内存
    memset(tmp, 0, sizeof(Student) * count);

    // 为每个结构体的 address 成员分配内存
    for(i = 0; i < count; i++)
    
        tmp[i].address = (char *)malloc(20);
    

    // 通过间接赋值 设置返回值
    *array = tmp;

    return ret;


3、释放结构体内存 ( 释放内存时先释放 指针成员内存 然后再释放结构头内存 )


释放结构体内存 : 释放 结构体 内存时 , 要先释放 结构体变量 的 一级指针 成员的内存 , 然后再释放整个 结构体的 内存 ;

/**
 * @brief free_student 释放内存
 * @param array
 * @return
 */
int free_student(Student **array, int count)

    // 返回值
    int ret = 0;

    // 循环控制变量
    int i = 0;

    // 验证二级指针合法性
    if(array == NULL)
    
        ret = -1;
        return ret;
    

    // 释放 每个结构体的 address 成员分配内存
    for(i = 0; i < count; i++)
    
        free((*array)[i].address);
        (*array)[i].address = NULL;
    

    // 释放 结构体内存
    free(*array);

    // 指针置空 , 防止野指针
    *array = NULL;

    return ret;





二、完整代码示例



#include <stdio.h>
#include <stdlib.h>
#include <string.h>

/**
 * @brief The Student struct
 * 定义 结构体 数据类型 , 同时为该结构体类型声明 别名
 * 可以直接使用 别名 结构体变量名 声明结构体类型变量
 * 不需要在前面添加 struct 关键字
 */
typedef struct Student

    // 声明变量时 , 会自动分配这 5 字节内存
    // 赋值时 , 可以直接使用 = 赋值字符串
    char name[5];
    int age;
    int id;
    // 声明变量时 , 只会为 4 字节指针分配内存
    // 具体的 字符串内存 需要额外使用 malloc 申请内存
    // 赋值时 , 必须使用 strcpy 函数 , 向堆内存赋值
    char *address;
Student;

/**
 * @brief printf_struct_array 打印结构体数组
 * @param array 数组作为函数参数退化为指针
 * @param count 数组中的元素个数
 */
void printf_struct_array(Student *array, int count)

    // 循环控制变量
    int i = 0;

    // 验证数组合法性
    if(array == NULL)
    
        return;
    

    // 打印结构体数组中的 结构体 age 字段
    for(i = 0; i < count; i++)
    
        printf("Student age = %d\\n", array[i].age);
    


/**
 * @brief sort_struct_array 对结构体数组 按照年龄进行排序
 * @param array 结构体指针
 * @param count 结构体数组的元素个数
 */
void sort_struct_array(Student *array, int count)

    // 循环控制变量
    int i = 0, j = 0;
    // 学生年龄
    Student tmp;

    // 验证数组合法性
    if(array == NULL)
    
        return;
    

    // 排序
    for(i = 0; i < count; i++)
    
        for(j = i + 1; j < count; j++)
        
            if(array[i].age > array[j].age)
            
                tmp = array[i];
                array[i] = array[j];
                array[j] = tmp;
            
        
    


/**
 * @brief create_student 堆内存中分配内存
 * @param array 二级指针 , 指向结构体数组
 * @return
 */
int create_student(Student **array, int count)

    // 返回值
    int ret = 0;
    // 循环控制变量

    int i = 0;
    // 临时变量
    Student *tmp = NULL;

    // 验证二级指针合法性
    if(array == NULL)
    
        ret = -1;
        return ret;
    

    // 堆内存中申请内存
    tmp = (Student *)malloc(sizeof(Student) * count);

    // 初始化分配的内存
    memset(tmp, 0, sizeof(Student) * count);

    // 为每个结构体的 address 成员分配内存
    for(i = 0; i < count; i++)
    
        tmp[i].address = (char *)malloc(20);
    

    // 通过间接赋值 设置返回值
    *array = tmp;

    return ret;


/**
 * @brief free_student 释放内存
 * @param array
 * @return
 */
int free_student(Student **array, int count)

    // 返回值
    int ret = 0;

    // 循环控制变量
    int i = 0;

    // 验证二级指针合法性
    if(array == NULL)
    
        ret = -1;
        return ret;
    

    // 释放 每个结构体的 address 成员分配内存
    for(i = 0; i < count; i++)
    
        free((*array)[i].address);
        (*array)[i].address = NULL;
    

    // 释放 结构体内存
    free(*array);

    // 指针置空 , 防止野指针
    *array = NULL;

    return ret;



/**
 * @brief 主函数入口
 * @return
 */
int main(int argc, char* argv[], char**env)


    // 声明结构体数组 , 该数组在栈内存中
    Student *array = NULL;
    // 循环控制变量
    int i = 0;

    // 堆内存中为结构体指针分配内存
    create_student(&array, 2);

    // 命令行中 , 接收输入的年龄
    for(i = 0; i < 2; i++)
    
        // 命令换行中 接收 输入的年龄 ,
        // 设置到 Student 数组元素的 age 成员中
        printf("\\n Input Age :\\n");
        scanf("%d", &(array[i].age));

        printf("\\n Input ID :\\n");
        scanf("%d", &(array[i].id));

        printf("\\n Input Name :\\n");
        scanf("%s", array[i].name);

        printf("\\n Input Address :\\n");
        scanf("%s", array[i].address);
    

    // 结构体数组 按照 age 排序
    sort_struct_array(array, 2);

    // 打印结构体数组中的 结构体 age 字段
    printf_struct_array(array, 2);

    // 释放堆内存数据
    free_student(&array, 2);

    // 命令行不要退出
    system("pause");
    return 0;


执行结果 :


 Input Age :
21

 Input ID :
1

 Input Name :
Tom

 Input Address :
China

 Input Age :
18

 Input ID :
2

 Input Name :
Jerry

 Input Address :
Russia
Student age = 18
Student age = 21
请按任意键继续. . .

以上是关于C语言,怎么为动态结构体数组分配内存的主要内容,如果未能解决你的问题,请参考以下文章

C语言结构体数组排序

C 语言结构体 ( 结构体中嵌套一级指针 | 分配内存时先 为结构体分配内存 然后再为指针分配内存 | 释放内存时先释放 指针成员内存 然后再释放结构头内存 )

在C语言中,怎么样定义结构体数组为全局变量?定义一个无返回值的函数,但是函数有参数可以吗?

C语言结构体数组的定义

C语言关于结构体变量为动态数组赋值问题

C语言中结构体数组名作为函数参数的问题