文章目录
一、二级指针案例 ( 返回自定义二级指针 | 精准控制内存大小 )
二、完整代码示例
一、二级指针案例 ( 返回自定义二级指针 | 精准控制内存大小 )
博客 【C 语言】二级指针案例 ( 字符串切割 | 返回 二维数组 作为结果 ) 中 , 使用 二维数组 , 接收字符串切割结果 ;
博客 【C 语言】二级指针案例 ( 字符串切割 | 返回 自定义二级指针 作为结果 ) 中 , 使用 自定义二级指针 , 接收字符串切割结果 ;
先分析出该 字符串中, 有多少个 逗号 字符 , 可以得到 二级指针 指向的 内存空间中 , 要存储多少 一级指针 , 也就是分析出有多少 行 , 然后在分析 每行 有多少列 , 即 为每个 一级指针 分配多少内存 ;
上述分配方式 , 能精准控制 内存 , 最大限度利用内存 ;
扫描 2 22 遍 ,
第一遍扫描 , 求出有多少个 一级指针 , 并为其分配内存 ;
第二次扫描 , 求出每个 一级指针 要分配多少内存 ;
第一次扫描 : 计算 要分割的字符串 个数 , 为其分配内存 ;
// 第一次遍历 , 求出有多少行 do { // 字符串中, 查找 字符 ',' // 找到后 , 返回 逗号 第一次出现的指针 p1 = strchr(p1, c); if (p1 != NULL) { // 将 p1 指针 与 p2 指针之间的 字符拷贝出来 // 这就是分割后的字符串 if (p1 - p2 > 0) { tmpcount ++; p1 = p2 = p1 + 1; } } else { // 如果 p1 为 NULL , 说明没有找到逗号字符 , 退出循环即可 break; } } while (*p1 != '\0'); // 得到分割的字符串个数 *count = tmpcount; // 为 一级指针 分配内存 p = (char **) malloc(tmpcount * sizeof(char *) ); if (myp == NULL) { return -1; } // 初始化分配的内存 memset(p, 0, tmpcount * sizeof(char *));
第二次扫描 : 为每个 一级指针 分配对应的内存 , 并拷贝 分割后的 字符串 ;
// 第二次遍历 // p1 , p2 初始化 p1 = str; p2 = str; tmpcount = 0; do { // 字符串中, 查找 字符 ',' // 找到后 , 返回 逗号 第一次出现的指针 p1 = strchr(p1, c); if (p1 != NULL) { // 将 p1 指针 与 p2 指针之间的 字符拷贝出来 // 这就是分割后的字符串 if (p1 - p2 > 0) { // 计算精准控制的 一级指针 指向的内存大小 int len = p1 - p2 + 1; // 为 一级指针 分配内存 p[tmpcount] = (char *) malloc(len * sizeof(char)); if(p[tmpcount] == NULL) { return -1; } // 将 p2 后的 p1 - p2 个字符 // 拷贝到数组中 strncpy(p[tmpcount], p2, p1 - p2); // 实际的字符拷贝完成后 , 再将 '\0' 字符拷贝过去 p[tmpcount][p1-p2] = '\0'; // 拷贝完成后 , 字符 tmpcount ++; // p2 和 p1 都设置为 指向 逗号后面字符 // 重新开始查找 逗号 并分割 p2 = p1 = p1 + 1; } } else { // 如果 p1 为 NULL , 说明没有找到逗号字符 , 退出循环即可 break; } } while (*p1 != '\0'); // 通过间接赋值 设置 分割后的字符串 到 自定义二级指针 *myp = p;
二、完整代码示例
完整代码示例 :
#include <stdio.h> #include <stdlib.h> #include <string.h> /** * @brief split_str 分割字符串 * @param str 要分割的字符串 * @param c 分割依据 * @param p 将分割结果写入该二级指针内存 * @param count 分割了多少份 * @return */ int split_str(const char *str, char c, char ***myp, int *count) { // 用于接收 str 参数 char *p1 = NULL, *p2 = NULL; // 临时变量 int tmpcount = 0; // 自定义的二级指针 char **p = NULL; // p1 , p2 初始化 p1 = str; p2 = str; // 第一次遍历 , 求出有多少行 do { // 字符串中, 查找 字符 ',' // 找到后 , 返回 逗号 第一次出现的指针 p1 = strchr(p1, c); if (p1 != NULL) { // 将 p1 指针 与 p2 指针之间的 字符拷贝出来 // 这就是分割后的字符串 if (p1 - p2 > 0) { tmpcount ++; p1 = p2 = p1 + 1; } } else { // 如果 p1 为 NULL , 说明没有找到逗号字符 , 退出循环即可 break; } } while (*p1 != '\0'); // 得到分割的字符串个数 *count = tmpcount; // 为 一级指针 分配内存 p = (char **) malloc(tmpcount * sizeof(char *) ); if (myp == NULL) { return -1; } // 初始化分配的内存 //memset(p, 0, tmpcount * sizeof(char *)); // 第二次遍历 // p1 , p2 初始化 p1 = str; p2 = str; tmpcount = 0; do { // 字符串中, 查找 字符 ',' // 找到后 , 返回 逗号 第一次出现的指针 p1 = strchr(p1, c); if (p1 != NULL) { // 将 p1 指针 与 p2 指针之间的 字符拷贝出来 // 这就是分割后的字符串 if (p1 - p2 > 0) { // 计算精准控制的 一级指针 指向的内存大小 int len = p1 - p2 + 1; // 为 一级指针 分配内存 p[tmpcount] = (char *) malloc(len * sizeof(char)); if(p[tmpcount] == NULL) { return -1; } // 将 p2 后的 p1 - p2 个字符 // 拷贝到数组中 strncpy(p[tmpcount], p2, p1 - p2); // 实际的字符拷贝完成后 , 再将 '\0' 字符拷贝过去 p[tmpcount][p1-p2] = '\0'; // 拷贝完成后 , 字符 tmpcount ++; // p2 和 p1 都设置为 指向 逗号后面字符 // 重新开始查找 逗号 并分割 p2 = p1 = p1 + 1; } } else { // 如果 p1 为 NULL , 说明没有找到逗号字符 , 退出循环即可 break; } } while (*p1 != '\0'); // 通过间接赋值 设置 分割后的字符串 到 自定义二级指针 *myp = p; return 0; } /** * @brief 主函数入口 * @return */ int main() { // 存放返回值 int ret = 0; // 字符串分割的份数 int count = 4; // 按照逗号分割 char split = ','; // 字符串 char *str = "12,ab,345,"; // 循环控制变量 int i = 0; // 存放分割结果, 此处自定义 二级指针 内存结构 char **p = NULL; // 分割字符串 ret = split_str(str, split, &p, &count); // split_str 方法执行失败 if(ret != 0) { printf("split error!\n"); return ret; } // 打印分割结果 for(i = 0; i < count; i++) { printf("%s\n", p[i]); } // 释放内存 // 先释放 num 个 一级指针 for(i = 0; i < count; i++) { if(p[i] != NULL) { free(p[i]); p[i] = NULL; } } // 再释放 二级指针 if(p != NULL) { free(p); } // 二维指针 置空 p = NULL; // 命令行不要退出 system("pause"); return 0; }
执行结果 :