一 (单)链表 ADT

+ Status InitList(LinkList &L) 初始化(单)链表

+ void printList(LinkList L) 遍历(单)链表

+ int ListLength(LinkList L) 获得表长

+ void CreateList_Head(LinkList &L, int n) 创建单链表 (头插法)【重点】

+ void CreateList_Rear(LinkList &L, int n) 创建单链表 (尾插法) 【重点】

+ Status GetElement(LinkList L, int i, ElementType &e) (按位)取值

+ LNode *LocateElement(LinkList L, ElementType e) (按值)查找

+ Status ListInsert(LinkList &L, int i, ElementType e) (按位)插入 【重点 | 易错】

+ Status ListDelete(LinkList &L, int i) (按位)删除  【重点 | 易错/易混】

二 编程实现

2.1 定义基础数据类型

ElementType (数据元素类型/结构体)

struct ElementType {
char data; // char -> ElementType bool operator==(const ElementType b) const{ // 重载结构体 ElementType 的运算符
return this->data == b.data;
} bool operator!=(const ElementType b) const{
return this->data != b.data;
}
};

Status (状态/枚举类型)

enum Status { ERROR, OK, OVERFLOW };

LNode(单链表结点/结构体) / LinkList (单链表/结构体)

typedef struct LNode { // typedef 关键字 可使用它来为类型取一个新的名称 like: typedef unsigned char BYTE;
ElementType data; //数据域
LNode *next; // 指针域
}*LinkList; // *LinkList 本质上等同于 *LNode

2.2 初始化(单)链表

Status InitList(LinkList &L)

Status InitList(LinkList &L){
L = new LNode; // 生成新结点 作为 头结点 , 头指针L指向头结点
L->next = NULL; // next指针 置空(挂起)
return OK;
}

2.3 遍历(单)链表 

void printList(LinkList L)

void printList(LinkList L){
LNode *p;
p = L->next;
printf("[LinkList.h#printList] List->elements: ");
while(p != NULL){
printElementType(p->data);
printf("\t");
p = p->next;
}
printf("\n");
}

2.4 获得表长

int ListLength(LinkList L)

int ListLength(LinkList L){
LNode *p = L->next; // 指针p 指向首元结点
int i = 0; //
while(p){ // 当前结点不为空
p = p->next;
i ++; // 长度 +1
}
return i;
}

2.5-1 创建单链表 (头插法)

void CreateList_Head(LinkList &L, int n)

void CreateList_Head(LinkList &L, int n){ // n: 要输入的元素个数
L = new LNode; // 生成头结点
L->next = NULL;
for(int i=0;i<n;i++){
LNode *s = new LNode; // 生成新结点
printf("\ninput -> element: ");
cin>>( s->data.data ); s->next = L->next; //链接 新结点
L->next = s;
}
}

2.5-2 创建单链表 (尾插法) 

void CreateList_Rear(LinkList &L, int n)

void CreateList_Rear(LinkList &L, int n){ // n: 要输入的元素个数
L = new LNode;
L->next = NULL; LNode *rear = L; // 初始化 尾指针 rear 指向 头结点
for(int j=0; j<n; j++){
LNode *s = new LNode; // 生成新结点
printf("\ninput -> element: ");
cin>>s->data.data; s->next = rear->next; //链接 新结点
rear->next = s; rear = rear->next; // 尾指针 rear 指向 新生成的尾结点
}
}

2.6 (按位)取值

Status GetElement(LinkList L, int i, ElementType &e)

Status GetElement(LinkList L, int i, ElementType &e){
LNode *p = L->next; // 指针p 指向首元结点
int j = 1;
while(p != NULL && j<i){
p = p->next;
j++;
}
if(j>i || p==NULL){ // i值不合法 i>n 或 i<0
return ERROR;
}
e = p->data; // 赋值
return OK;
}

2.7 (按值)查找 

LNode *LocateElement(LinkList L, ElementType e)

LNode *LocateElement(LinkList L, ElementType e){
LNode *p = L->next; // 指针p 指向首元结点
while(p!=NULL && p->data!=e){
p = p->next;
}
return NULL;
}

2.8 (按位)插入

Status ListInsert(LinkList &L, int i, ElementType e)

Status ListInsert(LinkList &L, int i, ElementType e){
LNode *p = L; // 指针p 指向头结点 【易错: 必须是头结点, 如果初始化p指向 首元结点(p = L->next), 那么 当i=1时, 会与 if(j>i-1 || p == NULL)中的"p == NULL"自相矛盾】
int j = 0; // j 表示下标(明确j的意义很重要, 到底是 表示 下标(从0开始) 还是位置序号(从1开始) )
while(j<i-1 && p!= NULL){ // 移动指针p 指向 第 i-1 个元素上 (即 使得 下标 j 最大等于 i-2)
p = p->next;
j++;
}
if(j>i-1 || p == NULL){ // 判断 i 值 是否合法 i>n 或者 i<0
return ERROR;
}
LNode *s = new LNode; // 生成新结点
s->data = e;
s->next = p->next; // 链接 新结点 与 原第 i 个结点
p->next = s;
return OK;
}

2.9 (按位)删除 

Status ListDelete(LinkList &L, int i)

Status ListDelete(LinkList &L, int i){
LNode *p = L; //指针p 指向首元结点
int j = 0; // j 表示 下标 (j 用于 下一次的循环判断条件)
LNode *q; // 定位被删除结点
while( (p != NULL) && (j<i-1) ){ // 指针p 移动到 第 i-1 个元素上 (即 j 最大取值为: i-1 )
//【易混】 (p != NULL) 可以换成 (p->next != NULL),无任何影响,原因:设置该判断条件主要目的是 为了 指针p 能够继续往后移动
printf("(old) p->data: %c\n", p->data.data);
p = p->next;
printf("(new) p->data: %c\n--------------\n", p->data.data);
j++;
}
if( j > i-1 || p->next == NULL ){ // 【易错: p != NULL (考虑的因素是: i>n )】
//【易错】 此处的 p->next == NULL 的判断条件不能轻易换成 p == NULL 因为 必须满足 p 指向于 被删除的第 i 个 (非空)结点的前置结点
// 即 要满足 第 i 个结点非空 ,且 p 指向的是其前置结点 所以 只能为 p->next == NULL
printf("j:%d\n", j);
return ERROR;
}
q = p->next;
p->next = q->next;
delete q; // 释放被删除结点的空间
return OK;
}

三 测试运行(Main.cpp)

#include <stdio.h>
//#include <iostream.h>
#include <iostream>
using namespace std; #include "base.h"
#include "LinkList.h" int main(){
LinkList L; // *LinkList 本质上等同于 *LNode InitList(L); // 初始化(单)链表 L int i = ListLength(L);
printf("length: %d \n",i); ElementType e;
Status status; e.data = 'A';
status = ListInsert(L, 1, e);
printf("status (after insert) : %d\n", status);
printList(L); e.data = 'B';
status = ListInsert(L, 2, e);
printf("status (after insert) : %d\n", status);
printList(L); e.data = 'C';
status = ListInsert(L, 3, e);
printf("status (after insert) : %d\n", status);
printList(L); status = ListDelete(L, 0);
printf("status (after delete) : %d\n", status);
printList(L); status = ListDelete(L, 4);
printf("status (after delete) : %d\n", status);
printList(L); status = ListDelete(L, 3);
printf("status (after delete) : %d\n", status);
printList(L); CreateList_Rear(L, 5); // input: A B C D E
printList(L); return 0;
}

运行结果

length: 0
status (after insert) : 1
[LinkList.h#printList] List->elements: A
status (after insert) : 1
[LinkList.h#printList] List->elements: A B
status (after insert) : 1
[LinkList.h#printList] List->elements: A B C
j:0
status (after delete) : 0
[LinkList.h#printList] List->elements: A B C
(old) p->data:
(new) p->data: A
--------------
(old) p->data: A
(new) p->data: B
--------------
(old) p->data: B
(new) p->data: C
--------------
j:3
status (after delete) : 0
[LinkList.h#printList] List->elements: A B C
(old) p->data:
(new) p->data: A
--------------
(old) p->data: A
(new) p->data: B
--------------
status (after delete) : 1
[LinkList.h#printList] List->elements: A B input -> element: A input -> element: B input -> element: C input -> element: D input -> element: E
[LinkList.h#printList] List->elements: A B C D E

四 参考资料

1 《数据结构(C语言版 第二版)》.严蔚敏.李冬梅.吴伟民

[C++]数据结构:线性表之(单)链表的更多相关文章

  1. Java数据结构-线性表之单链表LinkedList

    线性表的链式存储结构,也称之为链式表,链表:链表的存储单元能够连续也能够不连续. 链表中的节点包括数据域和指针域.数据域为存储数据元素信息的域,指针域为存储直接后继位置(一般称为指针)的域. 注意一个 ...

  2. [数据结构 - 第3章] 线性表之单链表(C++实现)

    一.类定义 单链表类的定义如下: #ifndef SIGNALLIST_H #define SIGNALLIST_H typedef int ElemType; /* "ElemType类型 ...

  3. 线性表之单链表C++实现

    线性表之单链表 一.头文件:LinkedList.h //单链表是用一组任意的存储单元存放线性表的元素,这组单元可以是连续的也可以是不连续的,甚至可以是零散分布在内存中的任意位置. //单链表头文件 ...

  4. [数据结构-线性表1.2] 链表与 LinkedList<T>(.NET 源码学习)

    [数据结构-线性表1.2] 链表与 LinkedList<T> [注:本篇文章源码内容较少,分析度较浅,请酌情选择阅读] 关键词:链表(数据结构)    C#中的链表(源码)    可空类 ...

  5. 【Java】 大话数据结构(2) 线性表之单链表

    本文根据<大话数据结构>一书,实现了Java版的单链表. 每个结点中只包含一个指针域的链表,称为单链表. 单链表的结构如图所示: 单链表与顺序存储结构的对比: 实现程序: package ...

  6. 数据结构(java版)学习笔记(三)——线性表之单链表

    单链表的优点: 长度不固定,可以任意增删. 单链表的缺点: 存储密度小,因为每个数据元素,都需要额外存储一个指向下一元素的指针(双链表则需要两个指针). 要访问特定元素,只能从链表头开始,遍历到该元素 ...

  7. 续上文----线性表之单链表(C实现)

    本文绪上文线性表之顺序表(C实现) 本文将继续使用单链表实现线性表的另外一种存储结构.这种使用链表实现的存储结构在内存中是不连续的. C实现代码如下: #include<stdio.h> ...

  8. 线性表 (单链表、循环链表-python实现)

    一.线性表 线性表的定义: 线性表是具有相同数据类型的有限数据的序列. 线性表的特点: 出了第一个元素外,每个元素有且仅有一个直接前驱,除最后一个元素外有且只有一个后继. 线性表是一种逻辑结构,表示元 ...

  9. Java数据结构-线性表之静态链表

    静态链表的定义: 节点由一个一维数组和一个指针域组成,数组用来存放数据元素,而指针域里面的指针(又称游标)用来指向下一个节点的数组下标. 这种链表称之为静态链表. 链表中的数组第一个和最后一个位置须要 ...

随机推荐

  1. 【Jenkins】修改Ubuntu下的jenkins端口号

    jenkins安装目录:/var/lib/jenkins jenkins日志目录:/var/log/jenkins/jenkins.logjenkins默认配置:/etc/default/jenkin ...

  2. ubuntu---记录.简单一句话安装tf

    卸载 sudo pip3 uninstall tensorflow sudo pip3 uninstall tensorflow-gpu sudo pip3 uninstall tensorflow- ...

  3. 【二进制优化-多重背包】zznu-oj-2120 : 安详--如何用尽钱币打赏主播获得最大好感度

    2120 : 安详 题目描述 spring最近喜欢上了B站新秀主播,身为顿顿吃黄焖鸡的土豪,当然要过去打赏一番,但是spring还是喜欢精打细算,所以在打赏的时候,想要掏出有限的钱,获得主播的最大好感 ...

  4. serializers--嵌套关系作为字段来表示

    参考官网:https://www.django-rest-framework.org/api-guide/relations/#nested-relationships 先建立model class ...

  5. 快捷键IntelliJ IDEA For Mac

    http://www.cnblogs.com/wxd0108/p/5295017.html Mac键盘符号和修饰键说明 ⌘ Command ⇧ Shift ⌥ Option ⌃ Control ↩︎  ...

  6. Codeforces Round #551 (Div. 2) E. Serval and Snake (交互题)

    人生第一次交互题ac! 其实比较水 容易发现如果查询的矩阵里面包含一个端点,得到的值是奇数:否则是偶数. 所以只要花2*n次查询每一行和每一列,找出其中查询答案为奇数的行和列,就表示这一行有一个端点. ...

  7. Docker搭建Redis一主两从三哨兵

    作者:oscarwin juejin.im/post/5d26b03de51d454fa33b1960 这次实验准备了三台云主机,系统为Debian,ip分别为:35.236.172.131 ,35. ...

  8. Excel2013下拉框选择自动填充颜色

    图一写判断公式,图二选择应用范围.

  9. fastJson与jackson性能对比

    转载:https://blog.csdn.net/u013433821/article/details/82905222最近项目用到fastJson和jackson,为了决定到底弃用哪个,随手写了个测 ...

  10. js快速将字符串数组转化为数字数组(互换)

    1.数字数组转化为字符串数组 var arr = [1, 2, 3, 4, 5, 6, 7, 8, 9]; arr.map(String); //结果: ['1', '2', '3', '4', '5 ...