C语言创建和操作单链表数据结构的实例教程

 更新时间:2016年04月25日 15:09:37   作者:xubin341719  
这篇文章主要介绍了C语言创建和操作单链表数据结构的实例教程,讲解使用C语言实现链表结构时指针的使用,需要的朋友可以参考下

1,为什么要用到链表

数组作为存放同类数据的集合,给我们在程序设计时带来很多的方便,增加了灵活性。但数组也同样存在一些弊病。如数组的大小在定义时要事先规定,不能在程序中进行调整,这样一来,在程序设计中针对不同问题有时需要3 0个大小的数组,有时需要5 0个数组的大小,难于统一。我们只能够根据可能的最大需求来定义数组,常常会造成一定存储空间的浪费。

我们希望构造动态的数组,随时可以调整数组的大小,以满足不同问题的需要。链表就是我们需要的动态数组。它是在程序的执行过程中根据需要有数据存储就向系统要求申请存储空间,决不构成对存储区的浪费。

链表是一种复杂的数据结构,其数据之间的相互关系使链表分成三种:单链表、循环链表、双向链表,下面将逐一介绍。

2,单向链表

单链表有一个头节点head,指向链表在内存的首地址。链表中的每一个节点的数据类型为结构体类型,节点有两个成员:整型成员(实际需要保存的数据)和指向下一个结构体类型节点的指针即下一个节点的地址(事实上,此单链表是用于存放整型数据的动态数组)。链表按此结构对各节点的访问需从链表的头找起,后续节点的地址由当前节点给出。无论在表中访问那一个节点,都需要从链表的头开始,顺序向后查找。链表的尾节点由于无后续节点,其指针域为空,写作为NULL。

如图所示:

2016425150545423.jpg (600×130)

上图还给出这样一层含义,链表中的各节点在内存的存储地址不是连续的,其各节点的地址是在需要时向系统申请分配的,系统根据内存的当前情况,既可以连续分配地址,也可以跳跃式分配地址。

3,单向链表程序的实现
(1),链表节点的数据结构定义

struct node 
{ 
  int num; 
  struct node *p; 
} ; 

在链表节点的定义中,除一个整型的成员外,成员p是指向与节点类型完全相同的指针。

在链表节点的数据结构中,非常特殊的一点就是结构体内的指针域的数据类型使用了未定义成功的数据类型。这是在C中唯一规定可以先使用后定义的数据结构。

(2),链表的创建、输出步骤
单链表的创建过程有以下几步:

1 ) 定义链表的数据结构;

2 ) 创建一个空表;

3 ) 利用malloc ( )函数向系统申请分配一个节点;

4 ) 将新节点的指针成员赋值为空。若是空表,将新节点连接到表头;若是非空表,将新

节点接到表尾;

5 ) 判断一下是否有后续节点要接入链表,若有转到3 ),否则结束;

单链表的输出过程有以下几步

1) 找到表头;

2) 若是非空表,输出节点的值成员,是空表则退出;

3 ) 跟踪链表的增长,即找到下一个节点的地址;

4) 转到2 ).

(3),程序代码例子:
创建一个存放正整数单链表,输入0或小于0的数,结束创建链表,并打印出链表中的值,程序如下:

#include <stdlib.h> /*含ma l l o c ( ) 的头文件*/ 
#include <stdio.h> 
 //①定义链表数据结构 
struct node 
{ 
  int num; 
  struct node *next; 
}; 
//函数声明 
struct node *creat();  
void print(); 
main( ) 
{ 
 
  struct node *head; 
  head=NULL;  //②建一个空表 
  head=creat(head);/*创建单链表*/ 
  print(head);/*打印单链表*/ 
} 
/******************************************/  
struct node*creat(struct node *head)/*返回的是与节点相同类型的指针*/ 
{ 
  struct node*p1,*p2; 
  int i=1; 
//③利用malloc ( )函数向系统申请分配一个节点 
  p1=p2=(struct node*)malloc(sizeof(struct node));/*新节点*/  
  printf("请输入值,值小于等于0结束,值存放地址为:p1_ADDR= %d\n",p1); 
  scanf("%d",&p1->num);/*输入节点的值*/ 
  p1->next=NULL;/*将新节点的指针置为空*/ 
  while(p1->num>0)/*输入节点的数值大于0*/ 
  { 
//④将新节点的指针成员赋值为空。若是空表,将新节点连接到表头;若是非空表,将新节点接到表尾;  
    if(head==NULL) 
      head=p1;/*空表,接入表头*/ 
    else  
      p2->next=p1;/*非空表,接到表尾*/ 
    p2=p1; 
 
    p1=(struct node*)malloc(sizeof(struct node));/*下一个新节点*/ 
    i=i+1; 
    printf("请输入值,值小于等于0结束,值存放地址为:p%d_ADDR= %d\n",i,p2); 
    scanf("%d",&p1->num);/*输入节点的值*/ 
//⑤判断一下是否有后续节点要接入链表,若有转到3 ),否则结束;  
  } 
//==============原来程序更正部分:(多谢@daling_datou提醒)================================ 
  free(p1); //申请到的没录入,所以释放掉  
  p1=NULL;  //使指向空  
  p2->next = NULL; //到表尾了,指向空  
  printf("链表输入结束(END)\n");  
//============================================== 
  return head;/*返回链表的头指针*/ 
} 
/*******************************************/ 
void print(struct node*head)/*出以head为头的链表各节点的值*/ 
{ 
  struct node *temp; 
  temp=head;/*取得链表的头指针*/ 
 
  printf("\n\n\n链表存入的值为:\n"); 
  while(temp!=NULL)/*只要是非空表*/ 
  { 
    printf("%6d\n",temp->num);/*输出链表节点的值*/ 
    temp=temp->next;/*跟踪链表增长*/ 
  } 
  printf("链表打印结束!!"); 
} 

2016425150704280.png (675×442)

在链表的创建过程中,链表的头指针是非常重要的参数。因为对链表的输出和查找都要从链表的头开始,所以链表创建成功后,要返回一个链表头节点的地址,即头指针。

程序执行流程:

2016425150725045.gif (450×391)

4,单链表操作基础示例

#include <stdio.h> 
#include <malloc.h> 
#define LEN sizeof(NODE) 
 
typedef struct _NODE//节点声明 
{ 
  int val; 
  struct _NODE* next; 
} NODE, *PNODE; 
 
void print(PNODE head){//打印所有节点 
  while (head) 
  { 
    printf("%3d",head->val); 
    head = head->next; 
  } 
  printf("\n"); 
} 
 
void insertHead(PNODE *pHead, int val){//头插法 
  PNODE n = (PNODE)malloc(LEN); 
  n->val = val; 
  n->next = *pHead; 
  *pHead = n; 
} 
 
void insertTail(PNODE *pHead, int val){//尾插法 
  PNODE t = *pHead; 
  PNODE n = (PNODE)malloc(LEN); 
  n->val = val; 
  n->next = NULL; 
  if (*pHead == NULL) 
  { 
    n->next = *pHead; 
    *pHead = n; 
  }else{ 
    while (t->next) 
    { 
      t = t->next; 
    } 
    t->next = n; 
  } 
} 
 
void deleteHead(PNODE *pHead){//删除头 
  if (*pHead == NULL) 
  { 
    return; 
  } 
  else 
  { 
    PNODE t = *pHead; 
    *pHead = (*pHead)->next; 
    free(t); 
  } 
} 
 
void deleteTail(PNODE *pHead){//删除尾 
  PNODE t = *pHead; 
  if (t == NULL) 
  { 
    return; 
  } 
  else if (t->next == NULL) 
  { 
    free(t); 
    *pHead = NULL; 
  } 
  else{     
    while (t->next->next != NULL) 
    { 
      t = t->next; 
    } 
    free(t->next); 
    t->next = NULL; 
  } 
} 
 
PNODE findByVal(PNODE head, int val){//根据值找到满足条件的第一个节点 
  while (head != NULL && head->val != val) 
  { 
    head = head->next; 
  } 
  return head; 
} 
 
PNODE findByIndex(PNODE head, int index){//根据索引找节点 
  if (index == 1) 
  { 
    return head; 
  } 
  else 
  { 
    int c = 1; 
    while (head != NULL && index != c) 
    { 
      head = head->next; 
      c++; 
    } 
  } 
  return head; 
} 
 
void insertByIndex(PNODE *pHead, int index, int val){//根据索引插入节点 
  if (index == 1) 
  { 
    insertHead(pHead, val); 
  } 
  else 
  { 
    PNODE t = findByIndex(*pHead,index - 1); 
    if (t == NULL) 
    { 
      return; 
    } 
    else 
    { 
      PNODE n = t->next; 
      t->next = (PNODE)malloc(LEN); 
      t->next->next = n; 
      t->next->val = val; 
    } 
  } 
} 
 
void deleteByIndex(PNODE *pHead, int index)//根据结点索引删除结点 
{ 
  if (index == 1) 
  { 
    deleteHead(pHead); 
  } 
  else 
  { 
    PNODE t= findByIndex(*pHead, index - 1); 
    if (t == NULL || t->next == NULL) 
    { 
      return; 
    }else{ 
      PNODE n = t->next->next; 
      free(t->next); 
      t->next = n; 
    } 
  } 
} 
 
void deleteByVal(PNODE *pHead, int val)//根据值删掉第一个满足条件的 
{ 
  if (*pHead == NULL)//如果空表退出 
  { 
    return; 
  } 
  else 
  { 
    if ((*pHead)->val == val)//如果第一个就是,删头 
    { 
      deleteHead(pHead); 
    } 
    else 
    { 
      PNODE t = *pHead; 
      while (t->next != NULL && t->next->val != val)//遍历,若t的next为空或者next是要找的节点则退出 
      { 
        t = t->next; 
      } 
      if (t->next)//如果t指向要找的结点的上一个节点 
      { 
        PNODE n = t->next->next;//删除要找的结点 
        free(t->next); 
        t->next = n; 
      } 
    } 
  } 
} 
 
void clear(PNODE *pHead)//清除链表 
{ 
  while ((*pHead) != NULL) 
  { 
    deleteHead(pHead);//从头删除 
  } 
} 
 
void main() 
{ 
  PNODE head = NULL; 
 
  insertTail(&head,1); 
  deleteHead(&head); 
  insertTail(&head,2); 
  insertTail(&head,3); 
  insertTail(&head,4); 
  insertTail(&head,5); 
  insertTail(&head,6); 
 
  print(head); 
  insertByIndex(&head, 6, 9); 
  print(head); 
  //deleteByIndex(&head,3); 
  deleteByVal(&head, 2); 
  print(head); 
  clear(&head); 
  print(head); 
  insertByIndex(&head,1,12); 
  print(head); 
} 

相关文章

  • C语言广播的使用详解

    C语言广播的使用详解

    顾名思义可以把自己的数据发送给在特定范围内的所有人;我们网络编程中的广播一般是通过特定的广播地址把自己的数据发送给局域网内当前在线的客户端
    2022-05-05
  • C语言学习之指针知识总结

    C语言学习之指针知识总结

    想突破C语言的学习,对指针的掌握是非常重要的,本文为大家总结了C语言中指针的相关知识点,文中的示例代码讲解详细,感兴趣的可以学习一下
    2022-07-07
  • 深度理解c++中的this指针

    深度理解c++中的this指针

    这篇文章主要介绍了C++编程指向成员的指针以及this指针的基本使用指南,与C语言一样,存储的数值被解释成为内存里的一个地址,需要的朋友可以参考下。
    2016-07-07
  • 浅析C语言位域和位段

    浅析C语言位域和位段

    以下是对C语言中的位域和位段进行了详细的分析介绍,需要的朋友可以过来参考下
    2013-08-08
  • 基于C++实现掷双骰游戏的示例代码

    基于C++实现掷双骰游戏的示例代码

    在最流行的博彩游戏中有一种名为“掷双骰”(craps)的骰子游戏,这种游戏在世界各地的娱乐场所和大街小巷非常受欢迎。本文将用C++语言实现这一游戏,需要的可以参考一下
    2022-03-03
  • C/C++中比较字符串的方法详解

    C/C++中比较字符串的方法详解

    这篇文章主要介绍了C/C++中比较字符串的方法,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2023-02-02
  • 浅谈返回函数内部new分配的内存的引用

    浅谈返回函数内部new分配的内存的引用

    下面小编就为大家带来一篇浅谈返回函数内部new分配的内存的引用。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2016-12-12
  • Qt spdlog日志模块的使用详解

    Qt spdlog日志模块的使用详解

    在Qt应用程序开发中,良好的日志系统至关重要,本文将介绍如何使用spdlog 1.5.0创建满足以下要求的日志系统,感兴趣的朋友一起看看吧
    2025-04-04
  • C语言实现YUV文件转JPEG格式

    C语言实现YUV文件转JPEG格式

    这篇文章主要为大家详细介绍了如何利用C语言实现将YUV文件转为JPEG格式,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下
    2023-12-12
  • OpenCV实现拼图板小游戏

    OpenCV实现拼图板小游戏

    这篇文章主要为大家详细介绍了OpenCV实现拼图板小游戏,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2019-05-05

最新评论