【C语言】指向指针的指针 char * *使用技巧

需求:

因为需要定义一个字符串数组,用来存放多个字符串,于是想到使用 char **这种结构——指向指针的指针,于是编写代码如下:

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

int main()
{
char * * ch;
char * p = "abcdsdfsdfsfsfsdfs";
char * q = "def";
ch=(char **)malloc(2*sizeof(char *));
*ch = p;
*(ch+1) = q;
printf("%s\n",**ch);

return EXIT_SUCCESS;
}

预想的是程序会输出 “abcdsdfsdfsfsfsdfs”这个字符串,因为:

由上面的图可以看出,*ch = p;这条语句执行之后,ch地址处存放的值就变成了 0x8000 即第一个字符串的地址,当执行到输出语句时:

printf("%s\n",**ch);
*ch指向的是0x8000,那么 **ch指向的就应该是 0x8000里面存放的字符串了。

这样理解之后,运行,结果惊讶!

Output:
Segmentation fault

提示段错误。这就奇了怪了。

仔细分析,这一定是内存错误,

不妨换一种输出格式,随便试试,于是改了代码如下:

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

int main()
{
char * * ch;
char * p = "abcdsdfsdfsfsfsdfs";
char * q = "def";
ch=(char **)malloc(2*sizeof(char *));
*ch = p;
*(ch+1) = q;

printf("%s\n%s\n%s\n",p,ch,*ch); //红色为修改的部分

return EXIT_SUCCESS;
}
此时,输出结果为:


abcdsdfsdfsfsfsdfs
���
abcdsdfsdfsfsfsdfs

明显是 p 和 *ch表示相同的值, 而 ch输出为乱码,应该表示的是地址! 
再换一种输出格式,全部都已地址,即数字的格式:

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

int main()
{
char * * ch;
char * p = "abcdsdfsdfsfsfsdfs";
char * q = "def";
ch=(char **)malloc(2*sizeof(char *));
*ch = p;
*(ch+1) = q;

printf("%d\n%d\n%d\n",p,ch,*ch);//红色表示修改的部分

return EXIT_SUCCESS;
}
输出结果为:
134514184
134520952
134514184
哦,原来这样,p和 *ch既能输出字符串又能输出地址。此时,结合现有成果和已有知识,猛然知道:

其实 %s 输出的字符串是,由参数表示的地址处的字符开始输出,直到 遇到字符串结束符‘\0’,而 %d只是输出地址。

并不是以前我以为的 输出的值相同只是格式不同 ,以前我以为 输出十进制数就是把字符串都变成了十进制数,这是多么严重的一个错误哇!

这样一切就清楚了。

总结:在编程序的时候,如果发现错了,但是,一时又找不出来哪里有问题的时候,我们不妨多试试,用相近的代码多多尝试,不一定哪次就来灵感了呢,而且,一定要剔除自己的成见,比如,我的这个程序里面 我想当然以为 输出的值都应该是相同的。多多练习,就会发现越来越多的问题,这样也能让自己的只是掌握的更加熟练,更加准确。还有就不不要懒,该编程运行看效果的时候就得编程,该画图标识的时候就得画图,另外,我觉得codepad在线编译器确实不错,极大的简便了编译过程,自从用了gcc之后,我就没用过 MS的 vc6.0 之类的了,windows平台的那种软件一天能死N次。

### C语言中的三级指针 #### 定义与解释 在C语言中,**三级指针**指向指针的指针指向另一个指针的数据结构。它本质上是一个指针变量,存储的是另一级指针的地址。通过这种机制可以间接访问更深层次的数据结构。 具体来说,如果有一个`char*** ptr`类型的三级指针,则其表示如下: - `ptr` 是一个指向 `char**` 类型数据的指针。 - `*ptr` 则是一个 `char**` 类型的指针,即指向 `char*` 的指针。 - `**ptr` 表示一个 `char*` 类型的指针,最终可以通过 `***ptr` 访问到实际的字符数据[^2]。 #### 使用场景 三级指针常用于动态分配二维或多维数组、函数返回复杂数据结构或者修改指针本身的内容等情况。它的主要作用在于能够传递并修改指针本身的值,而不仅仅是其所指向的内容。 --- ### 示例代码解析 以下是基于引用内容的一个典型例子来展示如何定义和操作三级指针: ```c #include <stdio.h> #include <stdlib.h> // 函数声明:传入三级指针以及子节点数量 int status1_read(char ***node, int *subnum) { const char *temp[] = {"status1", "value1", "123", "value2", "456"}; int i; // 设置子节点数为2 *subnum = 2; // 动态分配内存给二级指针数组 *node = (char **)malloc(3 * sizeof(char *)); // 将临时字符串赋值给新分配的空间 for (i = 0; i < 3; i++) { (*node)[i] = (char *)malloc((strlen(temp[i]) + 1) * sizeof(char)); strcpy((*node)[i], temp[i]); } return 0; } void status1_read_test() { char **strs; int num, i; // 调用函数初始化三级指针及其关联数据 status1_read(&strs, &num); // 打印结果 for (i = 0; i <= num; i++) { printf("%s\n", strs[i]); } // 清理资源 for (i = 0; i < 3; i++) { free(strs[i]); } free(strs); } int main() { status1_read_test(); return 0; } ``` #### 代码说明 1. **函数参数设计** - 参数 `char ***node` 是一个三级指针,允许函数内部对其所指向的对象重新分配空间或改变其值。 - 参数 `int *subnum` 是一个整型指针,用于传出子节点的数量。 2. **动态内存管理** - 使用 `malloc()` 分配足够的内存以容纳多个字符串指针。 - 对于每个字符串,再次调用 `malloc()` 来复制原始字符串内容至新的堆区位置。 3. **释放内存** - 遍历整个二级指针数组逐一释放单个字符串占用的内存区域。 - 最终释放由 `malloc()` 创建的一级指针数组。 --- ### 注意事项 - 在使用多级指针时需特别注意内存泄漏问题,确保每次申请的内存都能被正确回收。 - 处理复杂的嵌套指针逻辑容易引发错误,因此建议编写清晰易读的辅助函数简化流程[^2]。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值