C语言中指针变量为何如此容易掉进陷阱,如何避免这些深不见底的坑呢?

2026-04-18 22:263阅读0评论SEO资源
  • 内容介绍
  • 文章标签
  • 相关推荐

本文共计698个文字,预计阅读时间需要3分钟。

C语言中指针变量为何如此容易掉进陷阱,如何避免这些深不见底的坑呢?

首先,我们来看一个初始化带头结点的单链表的例子,LNode是节点变量,LinkList是链表变量。

ctypedef struct LNode { int data; // 数据域 struct LNode *next; // 指针域,指向下一个节点} LNode, *LinkList;

// 初始化链表void InitList(LinkList *L) { *L=(LNode *)malloc(sizeof(LNode)); // 分配头结点空间 if (!*L) exit(1); // 内存分配失败 (*L)->next=NULL; // 头结点的指针域为空}

先看一个初始化带头结点单链表的例子,LNode是结点变量,LinkList是结点指针变量,等同于LNode*

typedef struct LNode{ // 定义单链表节点类型 int data; struct LNode *next; }LNode,*LinkList;

例1、错误的方法:初始化带头结点的单链表

void InitList(LinkList L) { L = (LinkList)malloc(sizeof(LNode)); L->data = 3; L->next = NULL; } void Empty(LinkList L) { if(NULL == L) { printf("链表不存在\n"); } else if(NULL == L->next) { printf("空表\n"); } } int main() { LNode p; p.data = 2; printf("p.data = %d\n",p.data); LinkList L = NULL; InitList(L); printf("L->data = %d\n",L->data); return 0; }

输出结果

p.data = 2
链表不存在

例2、正确的方法:初始化带头结点的单链表

void InitList(LinkList *L) { (*L) = (LinkList)malloc(sizeof(LNode)); (*L)->data = 3; (*L)->next = NULL; } void Empty(LinkList L) { if(NULL == L) { printf("链表不存在\n"); } else if(NULL == L->next) { printf("空表\n"); } } int main() { LNode p; p.data = 2; printf("p.data = %d\n",p.data); LinkList L = NULL; InitList(&L); printf("L->data = %d\n",L->data); return 0; }

输出结果

p.data = 2
空表
L->data = 3

为什么第一个 InitList(LinkList L)函数不能初始化单链表L呢?不是用的地址传递吗?

答:

问题就在于L = (LinkList)malloc(sizeof(LNode));

这里的 L 是InitList函数的形参,也就是局部变量,函数运行结束后就不存在了。

而main函数中的 L 指针变量,实际上是存储在全局变量中的,两个L不是同一个L。

例1中,main函数中的 LinkList L 一开始是指向 NULL的

从InitList(L)执行开始,虽然InitList函数中也是对L进行操作,但是此时的L 是InitList函数中的形参L ,之后的也都是对于形参的操作,如下图

C语言中指针变量为何如此容易掉进陷阱,如何避免这些深不见底的坑呢?

例1 中传入的是LinkList 变量L,属于一级指针,不能对实参指针变量 L本身进行操作,来修改实参 L 的值,使得L指向新分配的LNode,所以实参 L 还是指向NULL,因为输出结果“链表不存在”。

例2中,传入的是LinkList 变量的指针 &L,属于二级指针,可以对实参指针变量 L 本身进行操作,来修改实参 L 的值,使得L指向新分配的LNode,(*L) = (LinkList)malloc(sizeof(LNode));

总结

到此这篇关于C语言中指针变量坑的文章就介绍到这了,更多相关C语言指针变量内容请搜索自由互联以前的文章或继续浏览下面的相关文章希望大家以后多多支持自由互联!

标签:先看

本文共计698个文字,预计阅读时间需要3分钟。

C语言中指针变量为何如此容易掉进陷阱,如何避免这些深不见底的坑呢?

首先,我们来看一个初始化带头结点的单链表的例子,LNode是节点变量,LinkList是链表变量。

ctypedef struct LNode { int data; // 数据域 struct LNode *next; // 指针域,指向下一个节点} LNode, *LinkList;

// 初始化链表void InitList(LinkList *L) { *L=(LNode *)malloc(sizeof(LNode)); // 分配头结点空间 if (!*L) exit(1); // 内存分配失败 (*L)->next=NULL; // 头结点的指针域为空}

先看一个初始化带头结点单链表的例子,LNode是结点变量,LinkList是结点指针变量,等同于LNode*

typedef struct LNode{ // 定义单链表节点类型 int data; struct LNode *next; }LNode,*LinkList;

例1、错误的方法:初始化带头结点的单链表

void InitList(LinkList L) { L = (LinkList)malloc(sizeof(LNode)); L->data = 3; L->next = NULL; } void Empty(LinkList L) { if(NULL == L) { printf("链表不存在\n"); } else if(NULL == L->next) { printf("空表\n"); } } int main() { LNode p; p.data = 2; printf("p.data = %d\n",p.data); LinkList L = NULL; InitList(L); printf("L->data = %d\n",L->data); return 0; }

输出结果

p.data = 2
链表不存在

例2、正确的方法:初始化带头结点的单链表

void InitList(LinkList *L) { (*L) = (LinkList)malloc(sizeof(LNode)); (*L)->data = 3; (*L)->next = NULL; } void Empty(LinkList L) { if(NULL == L) { printf("链表不存在\n"); } else if(NULL == L->next) { printf("空表\n"); } } int main() { LNode p; p.data = 2; printf("p.data = %d\n",p.data); LinkList L = NULL; InitList(&L); printf("L->data = %d\n",L->data); return 0; }

输出结果

p.data = 2
空表
L->data = 3

为什么第一个 InitList(LinkList L)函数不能初始化单链表L呢?不是用的地址传递吗?

答:

问题就在于L = (LinkList)malloc(sizeof(LNode));

这里的 L 是InitList函数的形参,也就是局部变量,函数运行结束后就不存在了。

而main函数中的 L 指针变量,实际上是存储在全局变量中的,两个L不是同一个L。

例1中,main函数中的 LinkList L 一开始是指向 NULL的

从InitList(L)执行开始,虽然InitList函数中也是对L进行操作,但是此时的L 是InitList函数中的形参L ,之后的也都是对于形参的操作,如下图

C语言中指针变量为何如此容易掉进陷阱,如何避免这些深不见底的坑呢?

例1 中传入的是LinkList 变量L,属于一级指针,不能对实参指针变量 L本身进行操作,来修改实参 L 的值,使得L指向新分配的LNode,所以实参 L 还是指向NULL,因为输出结果“链表不存在”。

例2中,传入的是LinkList 变量的指针 &L,属于二级指针,可以对实参指针变量 L 本身进行操作,来修改实参 L 的值,使得L指向新分配的LNode,(*L) = (LinkList)malloc(sizeof(LNode));

总结

到此这篇关于C语言中指针变量坑的文章就介绍到这了,更多相关C语言指针变量内容请搜索自由互联以前的文章或继续浏览下面的相关文章希望大家以后多多支持自由互联!

标签:先看