单链表ADT
本博客第一篇学术性博客,所以还是写点什么东西;
首先这篇博客以及以后的博客中的代码尽量百分之90是自己写过的;
可能有部分图片和代码是我认为别人更好的故摘抄下来,
本人三观正确,所以一定会表明来源;
—————————华丽的分割线——————————————
参考书籍——《数据结构于算法分析(C语言描述)》
链表是最基本的数据结构之一,当我们学习完C语言后就会涉及到一点点链表的知识;
本篇博客主要记录链表的一些简单思路以及它的一些主要例程;
按照c的约定我们先将链表的函数原型以及一些typedef定义都放在一个Lish.h头文件里
List.h:
#ifndef LIST_H
#define LIST_H typedef char ElementType; struct Node;
typedef struct Node *PtrToNode;
typedef PtrToNode Position;
typedef PtrToNode List; List MakeEmpty(List L);
int IsLast(Position P, List L);
int IsEmpty(List L);
Position FindPrevious(List L, ElementType x);
Position Find(List L, ElementType x);
void Delete(List L, ElementType x);
void Insert(List L, ElementType x, Position P);
void InsertToTail(List L, ElementType x);
void InsertToHead(List L, ElementType x);
void PrintList(List L);
void DeleteList(List L);
void Reverse(List L); #endif
本篇中的代码所建立的链表都是带有头节点的链表,而使用表头属于个人习惯,我们不再这里做深究
下面的函数尽量满足ADT的想法,没有写CreateList函数是因为该函数功能是对一个已经创建好了的链表的操作而不是一个纯练习的文件,这是个人对此的理解如有更好的解释请大方私信我!谢谢!
添加了两个Insert函数分别是InsertToTail和InsertToHead,从字面上来看就是尾插法和头插法,考虑到M.A.W的Insert例程需要传入位置P,而此举往往很麻烦,于是加上了一个直接插入到尾部和头部的函数
使整个ADT的想法更加完善。
操作图示:

这里主要将链表的一些主要例程写出来,并将这些函数封装在一个.c文件里增强复用性!
SingleLinkedList.c:
#include"List.h"
#include<stdio.h>
#include<stdlib.h> struct Node{
ElementType Element;
PtrToNode Next;
}; List MakeEmpty(List L)
{
if(NULL != L)
{
L->Next = NULL;
}
return L;
} int IsLast(Position P, List L)//L is unused
{
return P->Next == NULL;
} int IsEmpty(List L)
{
return L->Next == NULL;
} Position FindPrevious(List L, ElementType x)
{
Position P;
P = L;
while(P->Next && P->Next->Element != x)
P = P->Next;
return P;
} Position Find(List L, ElementType x)
{
Position P;
P = L->Next;
while(P && P->Element != x)
P = P->Next;
return P;
} void Delete(List L, ElementType x)
{
Position Pre, Tmpcell; Pre = FindPrevious(L, x);//we need find the previous of deleted element
if(!IsLast(Pre, L))
{
Tmpcell = Pre->Next;
Pre->Next = Tmpcell->Next;
free(Tmpcell);
}
} //we insert the element after the position p
void Insert(List L, ElementType x, Position P)
{
Position NewCell;
TmpCell = (List)malloc(sizeof(struct Node));
if(NULL == NewCell)
printf("No space for allocation!!");
else
{
NewCell->Element = x;
NewCell->Next = P->Next;
P->Next = NewCell;
}
}
//插入链表尾部(尾插法)
void InsertToTail(List L, ElementType x)
{
Position Last, NewCell;
Last = L;
/*遍历链表找到最后一个结点*/
while(NULL != Last->Next)
Last = Last->Next;
Insert(L, x, Last);
}
//插入链表头部(头插法)
void InsertToHead(List L, ElementType x)
{
Insert(L, x, L);
} void PrintList(List L)
{
PtrToNode Tmp;
Tmp = L->Next;
while(Tmp->Next)
{
printf("%c-", Tmp->Element);
Tmp = Tmp->Next;
}
printf("%c\n", Tmp->Element);
} void DeleteList(List L)
{
Position Tmp, P;
P = L->Next;
L->Next = NULL;
while(P != NULL)
{
Tmp = P->Next;
free(P);
P = Tmp;
}
free(L);
} void Reverse(List L)
{
Position CurrentPos, NextPos, PreviousPos; CurrentPos = L->Next;//当前单链表的第一个节点
PreviousPos = NULL;//指向新链表的第一个节点,假设开始为空
while(CurrentPos != NULL)
{
NextPos = CurrentPos->Next;//取得当前节点的下一个节点位置
CurrentPos->Next = PreviousPos;//当前节点连接成新的链表
PreviousPos = CurrentPos;
CurrentPos = NextPos;//遍历到下一个节点
}
L->Next = PreviousPos;//哑元节点连接新链表的头节点
}
//与上述思想差不多,主要在返回上
/*Asumming(假如)is no header and L is not empty*/
//List Reverse(List L)
//{
// Position CurrentPos, NextPos, PreviousPos;
//
// CurrentPos = L;
// PreviousPos = NULL;
// while(CurrentPos != NULL)
// {
// NextPos = CurrentPos->Next;
// CurrentPos->Next = PreviousPos;
// PreviousPos = CurrentPos;
// CurrentPos = NextPos;
// }
// return PreviousPos;
//}
//下面是复杂记忆写法
//void Reverse(List L)//含有头节点
//{
// Position Tmp, P;
// Tmp = L->Next;
// L->Next = NULL;
// while(Tmp != NULL)
// {
// P = Tmp->Next;
// Tmp->Next = L->Next;
// L->Next = Tmp;
// Tmp = P;
// }
//}
//List Reverse(List L)//不含头节点
//{
// PtrToNode Tmp, P;
// P = L;
// L = NULL;
// while(P != NULL){
// Tmp = P->Next;
// P->Next = L;
// L = P;
// P = Tmp;
// }
// return L;
//}
下面贴出自己写的一组测试代码
Test.c:
#include"List.h"
#include<stdio.h>
#include<stdlib.h> int main()
{
ElementType Elem, De, PreElem, Ins;
Position Tmp;
List L;
L = (List)malloc(sizeof(struct Node));
if(NULL == L)
printf("Allocation failure!!!");
L = MakeEmpty(L);
printf("Please enter the element until the end of '#':");
while((Elem = getchar()) != '#')
{
InsertToTail(L, Elem);
}
getchar();
PrintList(L);
//删除并输出
printf("Please enter the element you want to delete:");
scanf("%c", &De);
getchar(); Delete(L, De);
PrintList(L);
//插入并输出
printf("After which element do you want to insert:");
scanf("%c", &PreElem);
getchar();
Tmp = Find(L, PreElem); printf("What element do you want to insert:");
scanf("%c", &Ins);
getchar(); Insert(L, Ins, Tmp);
PrintList(L);
//将整个表倒置
Reverse(L);
printf("Now the reverse list is:");
PrintList(L);
//删除整个表
DeleteList(L);
return ;
}

单链表ADT的更多相关文章
- 数据结构:DHUOJ 单链表ADT模板应用算法设计:长整数加法运算(使用单链表存储计算结果)
		
单链表ADT模板应用算法设计:长整数加法运算(使用单链表存储计算结果) 时间限制: 1S类别: DS:线性表->线性表应用 题目描述: 输入范例: -5345646757684654765867 ...
 - [C++]数据结构:线性表之(单)链表
		
一 (单)链表 ADT + Status InitList(LinkList &L) 初始化(单)链表 + void printList(LinkList L) 遍历(单)链表 + int L ...
 - 单链表 C++ 实现 - 含虚拟头节点
		
本文例程下载链接:ListDemo 链表 vs 数组 链表和数组的最大区别在于链表不支持随机访问,不能像数组那样对任意一个(索引)位置的元素进行访问,而需要从头节点开始,一个一个往后访问直到查找到目标 ...
 - 动态单链表的传统存储方式和10种常见操作-C语言实现
		
顺序线性表的优点:方便存取(随机的),特点是物理位置和逻辑为主都是连续的(相邻).但是也有不足,比如:前面的插入和删除算法,需要移动大量元素,浪费时间,那么链式线性表 (简称链表) 就能解决这个问题. ...
 - python数据结构与算法之单链表
		
表的抽象数据类型 ADT list: #一个表的抽象数据类型 List(self) #表的构造操作,创建一个空表 is_empty ...
 - JAVA实现具有迭代器的线性表(单链表)
		
一,迭代器的基本知识: 1,为什么要用迭代器?(迭代:即对每一个元素进行一次“问候”) 比如说,我们定义了一个ADT(抽象数据类型),作为ADT的一种实现,如单链表.而单链表的基本操作中,大部分需要用 ...
 - 单链表-Python实现-jupyter->markdown 格式测试
		
单链表引入 顺序表 理解Python变量的本质: 变量存储的不是值,是值的地址 理解Python的 "="表示的是指向关系 案例: 交换a,b的值, a=10, b=20 a, b ...
 - C++泛化单链表
		
泛型单链表 单链表将每个数据分为节点,每个节点存储数据和指向下一个节点的指针.这样数据就不用在内存中使用连续的存储空间,有更大的灵活性. 这里将单链表分为节点类(Node)和链表类(singleLin ...
 - 时间复杂度分别为 O(n)和 O(1)的删除单链表结点的方法
		
有一个单链表,提供了头指针和一个结点指针,设计一个函数,在 O(1)时间内删除该结点指针指向的结点. 众所周知,链表无法随机存储,只能从头到尾去遍历整个链表,遇到目标节点之后删除之,这是最常规的思路和 ...
 
随机推荐
- uni-app编写小程序
			
今天项目还没有开始开发,网上看到了uni-app官网介绍可以一套代码,多端使用,感觉很厉害的样子,就做了个小程序页面试试. uni-app 使用 vue语法,小程序api,内嵌mpvue,使我们能够很 ...
 - Python day 02
			
基础&运算符 今日概要 循环 字符串格式化 运算符 编码 内容回顾 & 补充 内容回顾 计算机基础 解释器python 2 和 python 3 语法 print input if / ...
 - speech
			
1.李开复:一个人的成功,15%靠专业知识,其余15%人际沟通,公众演讲,以及影响他人的能力 2.演讲是一门遗憾的艺术 3.没有准备就等于准备失败 4.追求完美,就是在追求完蛋 5.宁可千日无机会,不 ...
 - python程序—封装案例
			
需求: 1.房子有户型.总面积.家具名称列表 房子没有任何家具 2.家具有名字和占地面积,其中 席梦思(bed):4平米 衣柜(chest): 2平米 餐桌(table): 1.5平米 3.将以上3个 ...
 - 手撸代码实现equals方法
			
重点都在注释里面写了,这里就不再重复叙述,贴上代码到博客主要是备用. package equals; class Book extends Object { private String title; ...
 - bilinear pooling
			
一.双线性汇合的计算过程: 第一步,计算Gram 矩阵: 对于一组H×W×D的feature maps,$\boldsymbol{x}_{i} \in \mathbb{R}^{D}$是图像的深度描述, ...
 - Spring boot 2.1.x 造成的问题
			
1. mysql-conector-java 版本默认使用了 8.0.1.5 导致:项目启动时的警告:Loading class `com.mysql.jdbc.Driver'. This is de ...
 - Git基本命令整理
			
git help <command> # 显示command的help git show # 显示某次提交的内容 git show $id git co -- <file> # ...
 - php(一)搭建php开发环境
			
1.下载php语言包 php作为一门语言,本身可以是一个纯绿色版的"文件夹"——称之为"php语言包".windows版的下载地址:https://window ...
 - ibm产品系列架构师技术路线