本博客第一篇学术性博客,所以还是写点什么东西;

首先这篇博客以及以后的博客中的代码尽量百分之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的更多相关文章

  1. 数据结构:DHUOJ 单链表ADT模板应用算法设计:长整数加法运算(使用单链表存储计算结果)

    单链表ADT模板应用算法设计:长整数加法运算(使用单链表存储计算结果) 时间限制: 1S类别: DS:线性表->线性表应用 题目描述: 输入范例: -5345646757684654765867 ...

  2. [C++]数据结构:线性表之(单)链表

    一 (单)链表 ADT + Status InitList(LinkList &L) 初始化(单)链表 + void printList(LinkList L) 遍历(单)链表 + int L ...

  3. 单链表 C++ 实现 - 含虚拟头节点

    本文例程下载链接:ListDemo 链表 vs 数组 链表和数组的最大区别在于链表不支持随机访问,不能像数组那样对任意一个(索引)位置的元素进行访问,而需要从头节点开始,一个一个往后访问直到查找到目标 ...

  4. 动态单链表的传统存储方式和10种常见操作-C语言实现

    顺序线性表的优点:方便存取(随机的),特点是物理位置和逻辑为主都是连续的(相邻).但是也有不足,比如:前面的插入和删除算法,需要移动大量元素,浪费时间,那么链式线性表 (简称链表) 就能解决这个问题. ...

  5. python数据结构与算法之单链表

    表的抽象数据类型 ADT list:                      #一个表的抽象数据类型 List(self)               #表的构造操作,创建一个空表 is_empty ...

  6. JAVA实现具有迭代器的线性表(单链表)

    一,迭代器的基本知识: 1,为什么要用迭代器?(迭代:即对每一个元素进行一次“问候”) 比如说,我们定义了一个ADT(抽象数据类型),作为ADT的一种实现,如单链表.而单链表的基本操作中,大部分需要用 ...

  7. 单链表-Python实现-jupyter->markdown 格式测试

    单链表引入 顺序表 理解Python变量的本质: 变量存储的不是值,是值的地址 理解Python的 "="表示的是指向关系 案例: 交换a,b的值, a=10, b=20 a, b ...

  8. C++泛化单链表

    泛型单链表 单链表将每个数据分为节点,每个节点存储数据和指向下一个节点的指针.这样数据就不用在内存中使用连续的存储空间,有更大的灵活性. 这里将单链表分为节点类(Node)和链表类(singleLin ...

  9. 时间复杂度分别为 O(n)和 O(1)的删除单链表结点的方法

    有一个单链表,提供了头指针和一个结点指针,设计一个函数,在 O(1)时间内删除该结点指针指向的结点. 众所周知,链表无法随机存储,只能从头到尾去遍历整个链表,遇到目标节点之后删除之,这是最常规的思路和 ...

随机推荐

  1. Codeforces Gym 101190M Mole Tunnels - 费用流

    题目传送门 传送门 题目大意 $m$只鼹鼠有$n$个巢穴,$n - 1$条长度为$1$的通道将它们连通且第$i(i > 1)$个巢穴与第$\left\lfloor \frac{i}{2}\rig ...

  2. 使用tp访问数据库时提示错误: 'PDO' not found, 和not defined constant mysql_attr_init_command 和call an undefined function Think\Template\simpleXml_load_string()函数

    第一个问题: PDO not found 是因为 php没有安装pdo扩展, 无法提供给 php 以 数据库访问功能, 所以 报错是在文件: Think/Db.class.php的里面. 解决方法是: ...

  3. 解决Ubuntu无法通过ssh远程登录问题

    1.  安装 open ssh: sudo apt-get install ssh sudo apt-get install openssh-server 2. 编辑配置文件,允许以 root 用户通 ...

  4. node中间层实现文件上传

    一般情况下,前端的文件上传一般都是通过form表单的(<input type="file" />)来完成文件的上传,如果使用node中间层完成跨域,文件的上传就需要在n ...

  5. Windows下安装Anaconda

    Windows下安装Anaconda   Anaconda介绍 Anaconda指的是一个开源的Python发行版本,其包含了conda.Python等180多个科学包及其依赖项.因为包含了大量的科学 ...

  6. vlog.hpp

    //vov #ifndef VLOG_HPP #define VLOG_HPP #include <sys/time.h> #include <unistd.h> #ifnde ...

  7. S3T mongodb GUI

    下载 cd ~/Downloads wget https://download.studio3t.com/studio-3t/linux/2019.2.1/studio-3t-linux-x64.ta ...

  8. git push origin master 上传失败

    http://blog.csdn.net/llf369477769/article/details/51917557 按照网上教程用git把项目上传到github,但是在最后一步git push or ...

  9. 如何系统地自学 Python?

    最近开始系统的学习Python,以及整理的一些资料.github记录着个人自学 Python 的过程,持续更新.欢迎大家一起来完善这个自学Python学习的项目,给后来者一个参考的学习过程.githu ...

  10. 20175317 MyCP(课下作业,必做)

    一.题目要求 编写MyCP.java 实现类似Linux下cp XXX1 XXX2的功能,要求MyCP支持两个参数: java MyCP -tx XXX1.txt XXX2.bin 用来把文本文件(内 ...