|
马上注册,结交更多好友,享用更多功能^_^
您需要 登录 才可以下载或查看,没有账号?立即注册
x
本帖最后由 Julia999 于 2019-8-1 16:18 编辑
离散存储【链表】定义:n个结点离散分配
彼此通过指针项链
每个结点只有一个前驱节点,一个后续节点
首节点没有前驱节点,尾节点没有后续节点
专业术语:
首节点:第一个有效的节点
尾节点:最后一个个有效的节点
头结点:第一个有效节点前面的节点,并不存放有效数据,加头结点的主要目的是方便对链表的操作
头节点的数据类型和首节点的一样
头指针:指向头结点的指针变量(存放头结点的地址)
尾指针:指向尾节点的指针
如果希望通过一个函数对链表进行处理,我们至少需要哪一些参数:
只需要一个:头指针,因为我们通过头指针,可以推算出链表的所有信息
- //构造节点
- typedef struct Node
- {
- int data; //数据域
- //指针指向的是跟自己本身类型相同的一个节点
- struct Node *pNext; //指针域
- }NODE, *PNODE; //NODE 等于与struct Node PNODE 等价于stru
复制代码
创建一个单链表并遍历:- #include<stdio.h>
- #include<malloc.h>
- #include<stdlib.h>
- //构造节点
- typedef struct Node
- {
- int data; //数据域
- //指针指向的是跟自己本身类型相同的一个节点
- struct Node *pNext; //指针域
- }NODE, *PNODE; //NODE 等于与struct Node PNODE 等价于struct Node*
- PNODE creat_list(void)
- {
- int len; //用来存放有效节点的个数
- int val; //用来临时存放节点的值
-
- //分配了一个不存放有效数据的头结点
- PNODE pHead=(PNODE)malloc(sizeof(NODE));
- if(NULL==pHead)
- {
- printf("分配失败!");
- exit(-1);
- }
-
- //使pTail永远指向尾节点
- PNODE pTail=pHead;
- pTail->pNext=NULL;
- printf("请输入你需要生成链表节点的个数:");
- scanf("%d",&len);
- //每循环一次,就用pNew生成了一个节点,并将节点的数据域赋值
- for(int i=0;i<len;i++)
- {
- printf("请输入第%d个节点的值");
- scanf("%d",&val);
- PNODE pNew=(PNODE)malloc(sizeod(NODE));
- if(NULL==pNew)
- {
- printf("分配失败!");
- exit(-1);
- }
- pNew->data=val;
- pTail->pNext=pNew;
- pNew->pNext=NULL;
- pTail=pNew;
- }
- return pHead;
- }
- void traverse_list(PNODE pHead) //遍历
- {
- //使p指向第一个有效节点
- PNODE p=pHead->next;
- while(NULL!=p)
- {
- //当链表不为空的时候
- printf("%d ",p->data);
- p=p->pNext;
- }
- printf("\n");
- }
- int main()
- {
- PNODE pHead=NULL; //等价于 struct Node *pHead=NULL
- pHead=creat_list(); //创建一个非循环单链表,并将该链表的头结点的地址赋给pHead
- traverse_list(pHead);
- return 0;
- }
复制代码
分类:
单链表
双链表:每一个节点有两个指针域
循环链表:能通过任何人一个节点找到其他所有的节点
非循环链表
算法:
遍历
查找
清空
销毁
求长度
排序
删除节点(删除p指向的节点之后的那一个节点)
- r=p->next; //必须将p->next给r,这样方便释放
- p->next=p->next->next; //p->next保存的是下一个节点的地址,也就是下一个节点,p->next->next表示的是下一个节点的指针域,也就是保存了下下个结点的地址,也就是下下个节点,直接将p的指针域指向下下个结点
- free(r);
复制代码 插入节点
(将q指向的节点放在p指向的节点之后)
- q->next=p->next; //将q的指针域指向p的下一个节点(q是一个指针变量,储存的是q的地址,所以q->next就是q的指针域,p->next存储的是p的下一个节点的地址,也就是p的下一个节点)
- p->next=q;
复制代码
对单链表的各种操作例子:
- #include<stdio.h>
- #include<malloc.h>
- #include<stdlib.h>
- //构造节点
- typedef struct Node
- {
- int data; //数据域
- //指针指向的是跟自己本身类型相同的一个节点
- struct Node *pNext; //指针域
- }NODE, *PNODE; //NODE 等于与struct Node PNODE 等价于struct Node*
- PNODE creat_list(void)
- {
- int len; //用来存放有效节点的个数
- int val; //用来临时存放节点的值
-
- //分配了一个不存放有效数据的头结点
- PNODE pHead=(PNODE)malloc(sizeof(NODE));
- if(NULL==pHead)
- {
- printf("分配失败!");
- exit(-1);
- }
-
- //使pTail永远指向尾节点
- PNODE pTail=pHead;
- pTail->pNext=NULL;
- printf("请输入你需要生成链表节点的个数:");
- scanf("%d",&len);
- //每循环一次,就用pNew生成了一个节点,并将节点的数据域赋值
- for(int i=0;i<len;i++)
- {
- printf("请输入第%d个节点的值");
- scanf("%d",&val);
- PNODE pNew=(PNODE)malloc(sizeod(NODE));
- if(NULL==pNew)
- {
- printf("分配失败!");
- exit(-1);
- }
- pNew->data=val;
- pTail->pNext=pNew;
- pNew->pNext=NULL;
- pTail=pNew;
- }
- return pHead;
- }
- void traverse_list(PNODE pHead) //遍历
- {
- //使p指向第一个有效节点
- PNODE p=pHead->next;
- while(NULL!=p)
- {
- //当链表不为空的时候
- printf("%d ",p->data);
- p=p->pNext;
- }
- printf("\n");
- }
- bool is_empty(PNODE pHead)
- {
- if(pHead->pNext==NULL)
- return true
- else
- return false;
- }
- int lenght_list(PNODE pHead)
- {
- PNODE p=pHead->pNext;
- int len=0;
- while(NULL!=p)
- {
- ++len;
- p=p->pNext;
- }
- return len;
- }
- //在pHead所指向链表的第pos个结点的前面插入一个新的节点,这个节点的值是val,并且pos的值是从1开始
- bool insert_list(PNODE pHead,int pos,int val)
- {
- int i=0;
- PNODE p=pHead;
- while(NULL!=p&&i<pos-1)
- {
- p=p->pNext;
- ++i;
- }
- if(i>pos-1||NULL=p)
- return false;
- PNODE pNew=(PNODE)malloc(sizeof(NODE));
- if(NULL=pNew)
- {
- printf("动态分配内存失败");
- exit(-1);
- }
- pNew->data=val;
- PNODE p=p->pNext;
- p->pNext=pNew;
- pNew->pNext=q;
- return true;
- }
- bool delete_list(PNODE pHead,int pos,int *pVal)
- {
- int i=0;
- PNODE p=pHead;
- while(NULL!=p->pNext&&i<pos-1)
- {
- p=p->pNext;
- ++i;
- }
- if(i>pos-1||NULL==p->pNext)
- return false;
- //保存要删的节点
- PNODE q=p->pNext;
- *pVal=q->data;
- //删除p节点后面的节点
- p->pNext=p->pNext->pNext;
- free(q);
- q=NULL;
- return true;
- }
- void sort_list(PNODE pHead)
- {
- int len=length_list(pHead);
- PNODE p,q;
- for(int i=0,p=pHead->pNext;i<len-1;i++,p=p->pNext)
- {
- for(int j=i+1,q=p->pNext;j<len;j++,q=q->qNext)
- {
- if(p->data>q->data)
- {
- t=p->data;
- p->data=q->data;
- q->data=t;
- }
- }
- }
- return;
- }
- int main()
- {
- int val;
- PNODE pHead=NULL; //等价于 struct Node *pHead=NULL
- pHead=creat_list(); //创建一个非循环单链表,并将该链表的头结点的地址赋给pHead
- traverse_list(pHead);
- sort_list(pHead);
- traverse_list(pHead);
- insert_list(pHead,4,33);
- traverse_list(pHead);
- delete_list(pHead,2,&val);
- traverse_list(pHead);
- return 0;
- }
复制代码
|
|