您可以捐助,支持我们的公益事业。

1元 10元 50元





认证码:  验证码,看不清楚?请点击刷新验证码 必填



  求知 文章 文库 Lib 视频 iPerson 课程 认证 咨询 工具 讲座 Modeler   Code  
会员   
 
   
 
 
     
   
 订阅
  捐助
[C/C++基础知识] 那些被遗忘的链表知识
 
作者 Eastmount的博客,火龙果软件    发布于 2014-06-17
 

最近快毕业了,复试又复习了一些知识.其中就包括那些被遗忘的链表知识,而它又是C语言中非常重要一个知识点.同时发现很多同学都会忘记该知识,所以通过这篇文章一方面帮助大家回忆链表知识,同时对刚接触C语言的同学也有帮助.我采用问答的方式回顾那些知识,希望能接受!
提示:该文章引用李凤霞(北理)的《C语言程序设计教程》及课件和谭浩强(清华)的《C程序设计》.

一.链表基本概念

1.什么是链表?

链表是一种常见的动态进行存储分配的数据结构.

2.为什么会出现链表这种结构呢?

(1).C语言中使用数组存放数据时,须先定义固定数组长度,确定元素个数.如果数据超过其容量就会发生数组溢出;为防止该溢出,往往会定义很大的数组,但这样又造成资源空间浪费.如果程序采用动态数组方法复制增长的数据,方法可行但效率太低;

(2).如果在数组中需要删除一个数据或插入一个数据时,此时需要将删除或插入点数组后面的数据依次移动,这样的移动也会导致程序效率非常低.

3.此时,链表这种动态存储数据的结构油然而生.

你是否看到了数组与链表两者一些简单区别呢?那么链表的基本单位又是什么呢?

结点是链表的基本存储单位,在链表中所有元素都存储在一个具有相同数据结构的结点中.一个结点对应一组数据元素,每个结点在内存中使用一块连续的存储空间(一个结点可由多种数据域组成),每个结点之间使用不连续的存储空间,结点之间通过指针链接.结点由数据域和指针域/链组成.常用定义如下:

struct node
{
dadatype data; //数据域
struct node *next; //指针域:指向node结点指针
};

4.知道了链表的基本存储单位后,那链表的基本组成部分是什么呢?

链表一般由三部分组成:

(1).表头指针:指向链表头结点的指针,头指针是链表的标志,通常用head定义头指针;

(2).表头结点:链表的第一个结点,一般不保存数据信息.链表中可没有表头结点(后面讲述),它是为方便引入结点.

(3).数据结点:实际保存数据信息的结点.示意图如下:

5.前面讲到可能链表中没有表头结点,那么链表常见形式有哪些呢?

常见的形式包括:有表头结点的单向链表、无表头结点的单向链表、有表头的单向循环表、无表头的单向循环表.其中有表头与无表头的差别在于是否有表头结点,插入删除操作对应不同的判断;单向链表与单向循环链表的区别在于最后一个数据结点指针是NULL还是指向表头结点.双向链表即两个指针分别指向前一个位置和后一个位置的链表.

6.那么链表中的常见操作包括哪些呢?

链表的常见操作包括:建立链表、遍历链表、求链表表长、插入数据、删除结点.下面将详细解决.

二.链表基本操作

1.建立链表

建立链表前先定义一个包含数据域与指针域的结构类型,然后建立指向表头结点的头指针head,通过malloc函数动态申请内存作为表头结点.其中void *malloc(int size)的头文件为"stdlib.h".动态分配长度size字节存储区.

//定义结构类型
typedef struct node
{
char name[20]; //数据域
struct node *next; //指针域
}NODE;
NODE *head,*p; //说明指针
//建立空链表(仅表头结点)
p=(NODE *)malloc(sizeof(NODE));
p->next=NULL;
head=p;
//插入一个数据结点
p=(NODE *)malloc(sizeof(NODE));
gets(p->name); //输入姓名
p->next=head->next; //p指向下一个结点=head指向下一个及诶单
head->next=p; //p结点插入表头结点head后

上面代码的执行过程如下图所示:

如果想通过函数实现建立链表的代码如下:

//建立n个结点的链表
void create(NODE *head,int n)
{
NODE *p;
for(;n>0;n--)
{
p=(NODE *)malloc(sizeof(NODE));
gets(p->name);
p->next=head->next;
head->next=p;
}
}

但是需要注意:通过此种方法建立时,总是在head后插入一个新的结点,这就导致最终插入的顺序为输入顺序的逆序存储该n个结点的信息.如果想顺序插入,只需要让head结点指向第一个插入结点p,第一个指向第二个,依次最后一个结点指向NULL即可.在约瑟夫循环中我将讲述.

2.遍历链表

遍历链表中某个结点,即从链表第一个结点开始依次进行查找通过output函数可以实现,如果想具体增加一些遍历条件可以在函数中添加,下面output函数依次输出学生姓名.

//遍历输出结果
void output(NODE *head)
{
NODE *p;
p=head->next; //含表头结点
while(p!=NULL)
{
puts(p->name);
p=p->next;
}
}

如果想计算链表的长度,如果含表头结点时,从第一个结点开始依次遍历,没找到一个结点其长度加1,直到链表尾.如果链表为空时,表头结点head->next==null.此时返回的为0即可.

//计算链表长度
int count(NODE *head)
{
int number=0;
NODE *p;
p=head->next;
while(p!=NULL)
{
number++;
p=p->next;
}
return number;
}

自定义main函数调用其函数,程序测试结果如下图所示,是不是反序一目了然.

3.插入数据

在链表中第i个结点后面插入一个新节点的算法如下:

(1).定位第i个结点.让指针q指向第i个结点,指针p指向要插入的结点.

(2).链接后面的指针:p->next=q->next.

(3).链接前面指针:q->next=p.

如下图所示过程:

具体代码如下图所示,其中采用insert函数插入新结点时,可能遇到两种特殊情况:其一是向空表中插入新节点,其二是向链表最后一个元素后面插入一个新结点.

//插入新结点 head头指针 p插入指针 i位置
void insert(NODE *head,NODE *p,int i)
{
NODE *q;
int n = 0;
q = head;
//第一步 寻找到第i个结点位置
while(n<i&&q->next!=NULL)
{
q = q->next; n++;
}
//第二步 链接后面的指针
p->next = q->next;
//第三步 链接前面的指针
q->next = p;
}

4.删除数据

在链表中可以删除任意一个数据结点,其中删除链表中第i个结点的算法如下:

(1).定位第i-1个结点位置.指针q指向第i-1个结点,指针p指向被删除结点.

(2).摘链:q->next=p->next.

(3).释放结点p:free(p).

其中void free(void *p)释放p所指向的内存空间,头文件为"stdlib.h".如下图所示:

具体代码如下图所示,同时通常在删除结点p后,需要把q指向的下一个新的结点赋值为p,可以继续执行删除操作.

//删除第i个结点
void delete_node(NODE *head,int i)
{
NODE *q,*p;
int n = 0;
q = head;
//第一步 寻找到第i-1个结点位置 q指针指向
while(n<i-1&&q->next!=NULL)
{
q = q->next;
n++;
}
if(q->next!=NULL)
{
//第二步 摘链 q指向p的下一个结点
p = q->next;
q->next = p->next;
//第三步 释放结点
free(p);
}
}

希望读者思考一个问题,在删除结点时,如果链指针摘链操作后没有free释放掉该结点,会导致什么结果呢?如果你学过C++或Jave,你又能回忆起它的内存管理和泄露知识吗?

三.链表经典问题-约瑟夫循环问题

通过上面对链表的讲述,你是否能回忆起一些它的简单知识呢?下面我想通过链表知识中最经典的题目"约瑟夫循环问题"让我们看看链表如何在实例中应用.

题目:有N个孩子围成一圈并依次编号(从1起),老师指定从第M个孩子开始报数,当报到第S个孩子时出列,然后下一个孩子从1开始继续报数,依次出列.求孩子出列的顺序或求最后一个孩子的编号.

输入:输入n(孩子个数),m(开始报数编号),s(报s出列).

输出:孩子出列顺序或最后一个孩子编号.

分析:如下图所示,当输入n=5,m=2,s=3时表示总共有5个孩子,通过单向循环链表围成一圈,m=2表示从第二个孩子开始报数,第二个孩子报数1,第三个报数2,第四个孩子报数3(s=3)出列.依次出列顺序为:4-2-1-3-5.

完成该程序需要:

(1).建立单向循环链表.注意此时建表是顺序建立,前面讲述的在head后插入新结点为逆序建表.此时需要依次插入head->a1->a2.最后在让q->next=head构建循环链表.(代码无表头结点)

(2).通过循环找到开始报数的结点,p指向开始报数的结点,q指向其前一个结点,因为删除p时需要通过前一个结点q摘链.

(3).循环依次报数删除结点,知道p=p->next退出循环,此时仅剩最后一个结点.

<span style="font-size:14px;">#include<stdio.h>
#include<stdlib.h>
//定义结构
typedef struct node
{
int no;
struct node * next;
}NODE;
int main()
{
int i,j,k;
int n,m,s; //n个孩子 从m个开始报数 s个出列
NODE *head,*p,*q; //头结点 p插入结点 q插入前一个结点
printf("请输入数字:\n");
scanf("%d %d %d",&n,&m,&s);
//建表 顺序插入无表头单向循环链表
head=NULL;
for(i=1;i<=n;i++) //i存储序列号
{
p=(NODE*)malloc(sizeof(NODE));
p->no=i;
if(head==NULL) head=p; //第一个结点存入head
else q->next=p; //q链接新插入结点p
q=p; //新插入结点构成链尾
}
q->next=head; //链尾链接链头构成循环
//寻找输出的位置m p为开始的结点 q为其前面一个结点
q=head;
p=head;
for(k=1;k<m;k++) //如果m=1 即第一个位置
{
p=p->next;
}
while(q->next!=p) //寻找q指针 q为p的前一个结点
{
q=q->next;
}
//删除结点及输出
printf("输出删除结点顺序:\n");
while(p->next!=p)
{
//寻找到要删除结点位置
for(j=1;j<s;j++)
{
q=p;
p=p->next;
}
//输出结点并删除
printf("%d ",p->no);
q->next=p->next;
free(p);
p=q->next;
}
printf("\n最后剩余结点:%d\n",p->no);
system("PAUSE");
return 0;
}</span>

测试用例及输出结果如下所示:

(1).输入n=5 m=2 s=3

(2).输入n=35 m=5 s=3

如果你是一位刚接触C语言的同学,希望文章能令你对链表有些认识;

如果你是考研或找工作的同学,希望对你在面试题或考研题中有所帮助;

如果你对链表有很深入的认识,希望当你阅读该文章时能对我这样的年轻人慧心一笑;

最后希望该文章对大家有所帮组,同时如果文章中有错误或不足之处,还请海涵!同时感谢母校BIT及老师,四年转瞬即逝,还有很多知识需要学习.这篇文章仅仅是自己对链表知识的一些总结及在线笔记,请尊重作者的劳动果实!

   
次浏览       
相关文章

深度解析:清理烂代码
如何编写出拥抱变化的代码
重构-使代码更简洁优美
团队项目开发"编码规范"系列文章
相关文档

重构-改善既有代码的设计
软件重构v2
代码整洁之道
高质量编程规范
相关课程

基于HTML5客户端、Web端的应用开发
HTML 5+CSS 开发
嵌入式C高质量编程
C++高级编程
   
Visual C++编程命名规则
任何时候都适用的20个C++技巧
C语言进阶
串口驱动分析
轻轻松松从C一路走到C++
C++编程思想
更多...   


C++并发处理+单元测试
C++程序开发
C++高级编程
C/C++开发
C++设计模式
C/C++单元测试


北京 嵌入式C高质量编程
中国航空 嵌入式C高质量编程
华为 C++高级编程
北京 C++高级编程
丹佛斯 C++高级编程
北大方正 C语言单元测试
罗克韦尔 C++单元测试
更多...