鱼C论坛

 找回密码
 立即注册
查看: 1199|回复: 0

[学习笔记] 非循环单链表及操作

[复制链接]
发表于 2019-8-1 16:18:22 | 显示全部楼层 |阅读模式

马上注册,结交更多好友,享用更多功能^_^

您需要 登录 才可以下载或查看,没有账号?立即注册

x
本帖最后由 Julia999 于 2019-8-1 16:18 编辑
离散存储【链表】定义:n个结点离散分配
           彼此通过指针项链
           每个结点只有一个前驱节点,一个后续节点
           首节点没有前驱节点,尾节点没有后续节点

专业术语:
首节点:第一个有效的节点
尾节点:最后一个个有效的节点
头结点:第一个有效节点前面的节点,并不存放有效数据,加头结点的主要目的是方便对链表的操作
头节点的数据类型和首节点的一样
头指针:指向头结点的指针变量(存放头结点的地址)
尾指针:指向尾节点的指针

如果希望通过一个函数对链表进行处理,我们至少需要哪一些参数:
只需要一个:头指针,因为我们通过头指针,可以推算出链表的所有信息
  1. //构造节点
  2. typedef struct Node
  3. {
  4.         int data;   //数据域
  5.         //指针指向的是跟自己本身类型相同的一个节点
  6.          struct Node *pNext;  //指针域
  7. }NODE, *PNODE;  //NODE 等于与struct Node  PNODE 等价于stru
复制代码


创建一个单链表并遍历
  1. #include<stdio.h>
  2. #include<malloc.h>
  3. #include<stdlib.h>

  4. //构造节点
  5. typedef struct Node
  6. {
  7.         int data;   //数据域
  8.         //指针指向的是跟自己本身类型相同的一个节点
  9.         struct Node *pNext;  //指针域
  10. }NODE, *PNODE;  //NODE 等于与struct Node  PNODE 等价于struct Node*

  11. PNODE creat_list(void)
  12. {
  13.         int len;  //用来存放有效节点的个数
  14.         int val; //用来临时存放节点的值
  15.    
  16.         //分配了一个不存放有效数据的头结点
  17.         PNODE pHead=(PNODE)malloc(sizeof(NODE));
  18.         if(NULL==pHead)
  19.         {
  20.                 printf("分配失败!");
  21.                 exit(-1);
  22.         }
  23.    

  24.         //使pTail永远指向尾节点
  25.         PNODE pTail=pHead;
  26.         pTail->pNext=NULL;

  27.         printf("请输入你需要生成链表节点的个数:");
  28.         scanf("%d",&len);

  29.         //每循环一次,就用pNew生成了一个节点,并将节点的数据域赋值
  30.         for(int i=0;i<len;i++)
  31.         {
  32.                 printf("请输入第%d个节点的值");
  33.                 scanf("%d",&val);

  34.                 PNODE pNew=(PNODE)malloc(sizeod(NODE));
  35.                 if(NULL==pNew)
  36.                 {
  37.                     printf("分配失败!");
  38.                 exit(-1);
  39.                 }
  40.                 pNew->data=val;

  41.                 pTail->pNext=pNew;
  42.                 pNew->pNext=NULL;
  43.                 pTail=pNew;

  44.         }
  45.         return pHead;
  46. }

  47. void traverse_list(PNODE pHead)  //遍历
  48. {
  49.         //使p指向第一个有效节点
  50.         PNODE p=pHead->next;
  51.         while(NULL!=p)
  52.         {
  53.                 //当链表不为空的时候
  54.                 printf("%d  ",p->data);
  55.                 p=p->pNext;
  56.         }
  57.         printf("\n");
  58. }

  59. int main()
  60. {
  61.         PNODE pHead=NULL;  //等价于 struct Node *pHead=NULL
  62.         pHead=creat_list();  //创建一个非循环单链表,并将该链表的头结点的地址赋给pHead   
  63.     traverse_list(pHead);

  64.         return 0;
  65. }
复制代码


分类:
单链表
双链表:每一个节点有两个指针域
循环链表:能通过任何人一个节点找到其他所有的节点
非循环链表


算法:
遍历
查找
清空
销毁
求长度
排序
删除节点(删除p指向的节点之后的那一个节点)
  1. r=p->next;   //必须将p->next给r,这样方便释放
  2.         p->next=p->next->next;  //p->next保存的是下一个节点的地址,也就是下一个节点,p->next->next表示的是下一个节点的指针域,也就是保存了下下个结点的地址,也就是下下个节点,直接将p的指针域指向下下个结点
  3.         free(r);
复制代码
插入节点
(将q指向的节点放在p指向的节点之后)
  1. q->next=p->next;  //将q的指针域指向p的下一个节点(q是一个指针变量,储存的是q的地址,所以q->next就是q的指针域,p->next存储的是p的下一个节点的地址,也就是p的下一个节点)
  2.         p->next=q;
复制代码

对单链表的各种操作例子:
  1. #include<stdio.h>
  2. #include<malloc.h>
  3. #include<stdlib.h>

  4. //构造节点
  5. typedef struct Node
  6. {
  7.         int data;   //数据域
  8.         //指针指向的是跟自己本身类型相同的一个节点
  9.         struct Node *pNext;  //指针域
  10. }NODE, *PNODE;  //NODE 等于与struct Node  PNODE 等价于struct Node*

  11. PNODE creat_list(void)
  12. {
  13.         int len;  //用来存放有效节点的个数
  14.         int val; //用来临时存放节点的值
  15.    
  16.         //分配了一个不存放有效数据的头结点
  17.         PNODE pHead=(PNODE)malloc(sizeof(NODE));
  18.         if(NULL==pHead)
  19.         {
  20.                 printf("分配失败!");
  21.                 exit(-1);
  22.         }
  23.    

  24.         //使pTail永远指向尾节点
  25.         PNODE pTail=pHead;
  26.         pTail->pNext=NULL;

  27.         printf("请输入你需要生成链表节点的个数:");
  28.         scanf("%d",&len);

  29.         //每循环一次,就用pNew生成了一个节点,并将节点的数据域赋值
  30.         for(int i=0;i<len;i++)
  31.         {
  32.                 printf("请输入第%d个节点的值");
  33.                 scanf("%d",&val);

  34.                 PNODE pNew=(PNODE)malloc(sizeod(NODE));
  35.                 if(NULL==pNew)
  36.                 {
  37.                     printf("分配失败!");
  38.                 exit(-1);
  39.                 }
  40.                 pNew->data=val;

  41.                 pTail->pNext=pNew;
  42.                 pNew->pNext=NULL;
  43.                 pTail=pNew;

  44.         }
  45.         return pHead;
  46. }

  47. void traverse_list(PNODE pHead)  //遍历
  48. {
  49.         //使p指向第一个有效节点
  50.         PNODE p=pHead->next;
  51.         while(NULL!=p)
  52.         {
  53.                 //当链表不为空的时候
  54.                 printf("%d  ",p->data);
  55.                 p=p->pNext;
  56.         }
  57.         printf("\n");
  58. }


  59. bool is_empty(PNODE pHead)
  60. {
  61.         if(pHead->pNext==NULL)
  62.                 return true
  63.         else
  64.         return false;
  65. }

  66. int lenght_list(PNODE pHead)
  67. {
  68.         PNODE p=pHead->pNext;
  69.         int len=0;
  70.         while(NULL!=p)
  71.         {
  72.                 ++len;
  73.                 p=p->pNext;
  74.         }
  75.         return len;
  76. }

  77. //在pHead所指向链表的第pos个结点的前面插入一个新的节点,这个节点的值是val,并且pos的值是从1开始
  78. bool insert_list(PNODE pHead,int pos,int val)
  79. {
  80.         int i=0;
  81.         PNODE p=pHead;
  82.         while(NULL!=p&&i<pos-1)
  83.         {
  84.                 p=p->pNext;
  85.                 ++i;
  86.         }
  87.         if(i>pos-1||NULL=p)
  88.                 return false;

  89.         PNODE pNew=(PNODE)malloc(sizeof(NODE));
  90.         if(NULL=pNew)
  91.         {
  92.                 printf("动态分配内存失败");
  93.                 exit(-1);
  94.         }
  95.         pNew->data=val;
  96.         PNODE p=p->pNext;
  97.         p->pNext=pNew;
  98.         pNew->pNext=q;

  99.         return true;

  100. }

  101. bool delete_list(PNODE pHead,int pos,int *pVal)
  102. {
  103.         int i=0;
  104.         PNODE p=pHead;
  105.         while(NULL!=p->pNext&&i<pos-1)
  106.         {
  107.                 p=p->pNext;
  108.                 ++i;
  109.         }
  110.         if(i>pos-1||NULL==p->pNext)
  111.                 return false;

  112.         //保存要删的节点
  113.         PNODE q=p->pNext;
  114.         *pVal=q->data;

  115.         //删除p节点后面的节点
  116.         p->pNext=p->pNext->pNext;
  117.         free(q);
  118.         q=NULL;

  119.         return true;

  120. }

  121. void sort_list(PNODE pHead)
  122. {
  123.         int len=length_list(pHead);
  124.         PNODE p,q;
  125.         for(int i=0,p=pHead->pNext;i<len-1;i++,p=p->pNext)
  126.         {
  127.                 for(int j=i+1,q=p->pNext;j<len;j++,q=q->qNext)
  128.                 {
  129.                         if(p->data>q->data)
  130.                         {
  131.                                 t=p->data;
  132.                                 p->data=q->data;
  133.                                 q->data=t;
  134.                         }

  135.                 }
  136.         }
  137.         return;

  138. }


  139. int main()
  140. {
  141.         int val;
  142.         PNODE pHead=NULL;  //等价于 struct Node *pHead=NULL
  143.         pHead=creat_list();  //创建一个非循环单链表,并将该链表的头结点的地址赋给pHead   
  144.     traverse_list(pHead);

  145.         sort_list(pHead);
  146.         traverse_list(pHead);

  147.         insert_list(pHead,4,33);
  148.         traverse_list(pHead);

  149.         delete_list(pHead,2,&val);
  150.         traverse_list(pHead);

  151.         return 0;
  152. }
复制代码




想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

小黑屋|手机版|Archiver|鱼C工作室 ( 粤ICP备18085999号-1 | 粤公网安备 44051102000585号)

GMT+8, 2024-4-29 01:21

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

快速回复 返回顶部 返回列表