【C 语言】结构体 ( 结构体中嵌套二级指针 | 为 结构体内的二级指针成员 分配内存 | 释放 结构体内的二级指针成员 内存 )

简介: 【C 语言】结构体 ( 结构体中嵌套二级指针 | 为 结构体内的二级指针成员 分配内存 | 释放 结构体内的二级指针成员 内存 )

文章目录

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

1、结构体中嵌套二级指针 类型声明

2、为 结构体内的二级指针成员 分配内存

3、释放 结构体内的二级指针成员 内存

二、完整代码示例





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



1、结构体中嵌套二级指针 类型声明


结构体中 嵌套 二级指针 , 二级指针 可以使用 指针数组 / 二维数组 / 自定义二级指针内存 三种内存模型的任意一种 ;


此处选择的模型是 自定义二级指针内存 ;



代码示例 :


/**
 * @brief The Student struct
 * 定义 结构体 数据类型 , 同时为该结构体类型声明 别名
 * 可以直接使用 别名 结构体变量名 声明结构体类型变量
 * 不需要在前面添加 struct 关键字
 */
typedef struct Student
{
    // 声明变量时 , 会自动分配这 5 字节内存
    // 赋值时 , 可以直接使用 = 赋值字符串
    char name[5];
    int age;
    int id;
    // 声明变量时 , 只会为 4 字节指针分配内存
    // 具体的 字符串内存 需要额外使用 malloc 申请内存
    // 赋值时 , 必须使用 strcpy 函数 , 向堆内存赋值
    char *address;
    // 学生小组成员 , 由多个字符串组成
    // 二级指针 , 指向多个 一级指针
    // 每个 一级指针 指向 一个字符串
    // 此处的 二级指针 可以使用 指针数组 / 二维数组 / 自定义二级指针内存
    // 此处选择的模型是 自定义二级指针内存
    char **team;
}Student;


2、为 结构体内的二级指针成员 分配内存


为二级指针成员分配内存时 , 先为二级指针分配内存 , 再为一级指针分配内存 ;


核心业务逻辑 :

// 为每个结构体的 address 成员分配内存
    for(i = 0; i < count; i++)
    {
        // 为一级指针分配内存模型
        tmp[i].address = (char *)malloc(20);
        // 为 二级指针 内存模型分配内存 , 分配 3 个 一级指针 变量内存
        char **p = (char **)malloc(3 * sizeof(char *));
        // 为 二级指针 指向的 一级指针 分配内存
        for(j = 0; j < 3; j++)
        {
            // 每个一级指针分配 10 字节数据
            p[j] = (char *)malloc(10 * sizeof(char));
        }
        // 将分配好内存的 二级指针 模型 , 赋值给结构体中的二级指针
        tmp[i].team = p;
    }



代码示例 :


/**
 * @brief create_student 堆内存中分配内存
 * 为二级指针成员分配内存时 , 先为二级指针分配内存 , 再为一级指针分配内存
 * @param array 二级指针 , 指向结构体数组
 * @return
 */
int create_student(Student **array, int count)
{
    // 返回值
    int ret = 0;
    // 循环控制变量
    int i = 0, j = 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);
        // 为 二级指针 内存模型分配内存 , 分配 3 个 一级指针 变量内存
        char **p = (char **)malloc(3 * sizeof(char *));
        // 为 二级指针 指向的 一级指针 分配内存
        for(j = 0; j < 3; j++)
        {
            // 每个一级指针分配 10 字节数据
            p[j] = (char *)malloc(10 * sizeof(char));
        }
        // 将分配好内存的 二级指针 模型 , 赋值给结构体中的二级指针
        tmp[i].team = p;
    }
    // 通过间接赋值 设置返回值
    *array = tmp;
    return ret;
}



3、释放 结构体内的二级指针成员 内存


释放内存时 , 先释放 二级指针 指向的 一级指针 的内存 , 再释放 二级指针 内存 ;


核心业务逻辑 :


 

// 释放 每个结构体的 address 成员分配内存
    for(i = 0; i < count; i++)
    {
        // 释放一级指针
        free((*array)[i].address);
        (*array)[i].address = NULL;
        // 释放二级指针指向的一级指针
        for(j = 0; j < 3; j++)
        {
            if((*array)[i].team[j] != NULL){
                free( (*array)[i].team[j] );
                (*array)[i].team[j] = NULL;
            }
        }
        // 释放二级指针
        if((*array)[i].team != NULL)
        {
            free( (*array)[i].team );
            (*array)[i].team = NULL;
        }
    }


代码示例 :


/**
 * @brief free_student 释放内存
 * 释放内存时 , 先释放 二级指针 指向的 一级指针 的内存 , 再释放 二级指针 内存
 * @param array
 * @return
 */
int free_student(Student **array, int count)
{
    // 返回值
    int ret = 0;
    // 循环控制变量
    int i = 0, j = 0;
    // 验证二级指针合法性
    if(array == NULL)
    {
        ret = -1;
        return ret;
    }
    // 释放 每个结构体的 address 成员分配内存
    for(i = 0; i < count; i++)
    {
        // 释放一级指针
        free((*array)[i].address);
        (*array)[i].address = NULL;
        // 释放二级指针指向的一级指针
        for(j = 0; j < 3; j++)
        {
            if((*array)[i].team[j] != NULL){
                free( (*array)[i].team[j] );
                (*array)[i].team[j] = NULL;
            }
        }
        // 释放二级指针
        if((*array)[i].team != NULL)
        {
            free( (*array)[i].team );
            (*array)[i].team = 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;
    // 学生小组成员 , 由多个字符串组成
    // 二级指针 , 指向多个 一级指针
    // 每个 一级指针 指向 一个字符串
    // 此处的 二级指针 可以使用 指针数组 / 二维数组 / 自定义二级指针内存
    // 此处选择的模型是 自定义二级指针内存
    char **team;
}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, j = 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);
        // 为 二级指针 内存模型分配内存 , 分配 3 个 一级指针 变量内存
        char **p = (char **)malloc(3 * sizeof(char *));
        // 为 二级指针 指向的 一级指针 分配内存
        for(j = 0; j < 3; j++)
        {
            // 每个一级指针分配 10 字节数据
            p[j] = (char *)malloc(10 * sizeof(char));
        }
        // 将分配好内存的 二级指针 模型 , 赋值给结构体中的二级指针
        tmp[i].team = p;
    }
    // 通过间接赋值 设置返回值
    *array = tmp;
    return ret;
}
/**
 * @brief free_student 释放内存
 * 释放内存时 , 先释放 二级指针 指向的 一级指针 的内存 , 再释放 二级指针 内存
 * @param array
 * @return
 */
int free_student(Student **array, int count)
{
    // 返回值
    int ret = 0;
    // 循环控制变量
    int i = 0, j = 0;
    // 验证二级指针合法性
    if(array == NULL)
    {
        ret = -1;
        return ret;
    }
    // 释放 每个结构体的 address 成员分配内存
    for(i = 0; i < count; i++)
    {
        // 释放一级指针
        free((*array)[i].address);
        (*array)[i].address = NULL;
        // 释放二级指针指向的一级指针
        for(j = 0; j < 3; j++)
        {
            if((*array)[i].team[j] != NULL){
                free( (*array)[i].team[j] );
                (*array)[i].team[j] = NULL;
            }
        }
        // 释放二级指针
        if((*array)[i].team != NULL)
        {
            free( (*array)[i].team );
            (*array)[i].team = 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);
        printf("\n Input Team 1 Name :\n");
        scanf("%s", array[i].team[0]);
        printf("\n Input Team 2 Name :\n");
        scanf("%s", array[i].team[1]);
        printf("\n Input Team 3 Name :\n");
        scanf("%s", array[i].team[2]);
    }
    // 结构体数组 按照 age 排序
    sort_struct_array(array, 2);
    // 打印结构体数组中的 结构体 age 字段
    printf_struct_array(array, 2);
    // 释放堆内存数据
    free_student(&array, 2);
    // 命令行不要退出
    system("pause");
    return 0;
}




执行结果 :



Input Age :
18
 Input ID :
1
 Input Name :
Tom
 Input Address :
CHina
 Input Team 1 Name :
Tom1
 Input Team 2 Name :
Tom2
 Input Team 3 Name :
Tom3
 Input Age :
16
 Input ID :
2
 Input Name :
Jerry
 Input Address :
Beijing
 Input Team 1 Name :
Jerry1
 Input Team 2 Name :
Jerry2
 Input Team 3 Name :
Jerry3
Student age = 16
Student age = 18
请按任意键继续. . .

image.png

目录
相关文章
|
6月前
|
存储 网络协议 编译器
【C语言】深入解析C语言结构体:定义、声明与高级应用实践
通过根据需求合理选择结构体定义和声明的放置位置,并灵活结合动态内存分配、内存优化和数据结构设计,可以显著提高代码的可维护性和运行效率。在实际开发中,建议遵循以下原则: - **模块化设计**:尽可能封装实现细节,减少模块间的耦合。 - **内存管理**:明确动态分配与释放的责任,防止资源泄漏。 - **优化顺序**:合理排列结构体成员以减少内存占用。
491 14
|
6月前
|
存储 编译器 C语言
【C语言】结构体详解 -《探索C语言的 “小宇宙” 》
结构体通过`struct`关键字定义。定义结构体时,需要指定结构体的名称以及结构体内部的成员变量。
356 10
|
7月前
|
存储 C语言
C语言如何使用结构体和指针来操作动态分配的内存
在C语言中,通过定义结构体并使用指向该结构体的指针,可以对动态分配的内存进行操作。首先利用 `malloc` 或 `calloc` 分配内存,然后通过指针访问和修改结构体成员,最后用 `free` 释放内存,实现资源的有效管理。
634 13
|
7月前
|
存储 编译器 数据处理
C 语言结构体与位域:高效数据组织与内存优化
C语言中的结构体与位域是实现高效数据组织和内存优化的重要工具。结构体允许将不同类型的数据组合成一个整体,而位域则进一步允许对结构体成员的位进行精细控制,以节省内存空间。两者结合使用,可在嵌入式系统等资源受限环境中发挥巨大作用。
212 12
|
7月前
|
存储 数据建模 程序员
C 语言结构体 —— 数据封装的利器
C语言结构体是一种用户自定义的数据类型,用于将不同类型的数据组合在一起,形成一个整体。它支持数据封装,便于管理和传递复杂数据,是程序设计中的重要工具。
|
7月前
|
存储 算法 程序员
C 语言指针详解 —— 内存操控的魔法棒
《C 语言指针详解》深入浅出地讲解了指针的概念、使用方法及其在内存操作中的重要作用,被誉为程序员手中的“内存操控魔法棒”。本书适合C语言初学者及希望深化理解指针机制的开发者阅读。
|
7月前
|
存储 C语言 计算机视觉
在C语言中指针数组和数组指针在动态内存分配中的应用
在C语言中,指针数组和数组指针均可用于动态内存分配。指针数组是数组的每个元素都是指针,可用于指向多个动态分配的内存块;数组指针则指向一个数组,可动态分配和管理大型数据结构。两者结合使用,灵活高效地管理内存。
|
7月前
|
容器
在使用指针数组进行动态内存分配时,如何避免内存泄漏
在使用指针数组进行动态内存分配时,避免内存泄漏的关键在于确保每个分配的内存块都能被正确释放。具体做法包括:1. 分配后立即检查是否成功;2. 使用完成后及时释放内存;3. 避免重复释放同一内存地址;4. 尽量使用智能指针或容器类管理内存。
|
5月前
|
存储 算法 C语言
【C语言程序设计——函数】素数判定(头歌实践教学平台习题)【合集】
本内容介绍了编写一个判断素数的子函数的任务,涵盖循环控制与跳转语句、算术运算符(%)、以及素数的概念。任务要求在主函数中输入整数并输出是否为素数的信息。相关知识包括 `for` 和 `while` 循环、`break` 和 `continue` 语句、取余运算符 `%` 的使用及素数定义、分布规律和应用场景。编程要求根据提示补充代码,测试说明提供了输入输出示例,最后给出通关代码和测试结果。 任务核心:编写判断素数的子函数并在主函数中调用,涉及循环结构和条件判断。
296 23
|
4月前
|
人工智能 Java 程序员
一文彻底搞清楚C语言的函数
本文介绍C语言函数:函数是程序模块化的工具,由函数头和函数体组成,涵盖定义、调用、参数传递及声明等内容。值传递确保实参不受影响,函数声明增强代码可读性。君志所向,一往无前!
87 1
一文彻底搞清楚C语言的函数