链表的相关基础操作

# include <iostream>
using namespace std; typedef struct LNode {
int data; //结点的数据域
struct LNode* next; //结点的指针域
}LNode, * LinkList; //LinkList为指向结构体LNode的指针类型
//typedef: struct LNode == LNode; struct LNode* == LinkList //链表的初始化
LinkList InitList() {
LNode* L = new LNode; //生成新节点作为头结点,用头指针L指向头结点
L->next = NULL; //头结点的指针域置空
return L;
} //创建链表:头插法
void CreateList_H(LinkList& L, int n) {
//逆位序输入n个元素的值,建立带头结点的单链表L
L = new LNode;
L->next = NULL; //先建立一个带头结点的单链表
for (int i = 0; i < n; i++) {
LNode* p = new LNode; //生成新结点
cin >> p->data; //输入元素值赋值给新结点的数据域
p->next = L->next;
L->next = p;
}
} //创建链表:尾插法
void CreateList_R(LinkList& L, int n) {
//正位序输入n个元素的值,建立带头节点的单链表L
L = new LNode;
L->next = NULL; //先建立一个带头结点的空链表
LNode* r = L; //尾指针初始时指向头结点
for (int i = 0; i < n; i++) {
LNode* p = new LNode; //生成新结点
cin >> p->data; //输入元素值赋值给新结点的数据域
p->next = NULL; //p是最后一个结点
r->next = p;
r = p;//r指向新的尾结点p
}
} //按序号查找结点的值
bool GetElem(LinkList L, int i, int& e) {
//在带头结点的单链表L中根据序号i获取元素的值,用e返回L中第i个数据元素
LNode* p = L->next; //p指向首元结点
int j = 1; //j为计数器
while (p && j < i) { //顺链域向后扫描,直到p指向第i个元素或者扫描完p为空
p = p->next; //p指向下一个结点
j++; //计数器j相应加1
}
if (!p || j > i) return false; //p为空 或者 i值不合法(例如:i > n 或者 i<= 0)
e = p->data; //e保存第i个结点的数据域
return true;
} //按值查找链表结点
bool LocateElem(LinkList L, int e) {
//在带头结点的单链表L中查找值为e的元素
LNode* p = L->next;
while (p && p->data != e) {//顺链域向后扫描,直到p为空或者p所指结点的数据域等于e
p = p->next; //p指向下一个结点
}
if (!p) return false;
return true;
} //单链表的插入
/*
和顺序表一样,如果表中有n个结点, 则插入操作中的合法位置有n + 1个
即1<= i <= n+1. 当i = n + 1时,新节点则插在链表尾部
*/
bool InsertList(LinkList& L, int i, int e) {
//在点头结点的单链表L中第i个位置插入值为e的新结点
LNode* p = L; //为啥不能是LNode* p = L->next; int j = 1
int j = 0;
while (p && j < i - 1) { //查找第i-1个结点,p指向该结点
p = p->next;
j++;
}
if (!p || j > i - 1) return false; //i值不合法(例如:i > n + 1 或 i < 1)
LNode* s = new LNode;
s->data = e;
s->next = p->next;
p->next = s;
return true;
} //单链表的删除
/*
删除算法中的循环条件(p->next && j < i - 1)和插入算法中的循环条件
(p && j < i - 1)是有所区别的。因为插入操作中合法的插入位置有n+1个,而
删除操作中删除的合法位置只有n个,如果使用与插入操作相同的循环条件,则
会出现引入空指针的情况,使删除操作失败。
插入时p指针最多指向第n个结点,此时判断条件为while(p && j < i - 1)
删除时p指针最多指向第n-1个结点,此时判断条件为while(p->next && j < i - 1)
*/
bool DeleteList(LinkList& L, int i) {
//在带头结点的单链表L中,删除第i个位置
LNode* p = L, * q;
int j = 0;
while (p->next && j < i - 1) { //查找第i-1个结点,p指向该结点
p = p->next;
j++;
}
if (!(p->next) || j > i - 1)return false; //当i>n+1 或者i < 1时,删除位置不合理
q = p->next; //临时保存被删结点的地址以备释放空间
p->next = q->next; //改变删除结点前驱结点的指针域
delete q; //释放被删除结点的空间
return true;
} //单链表的遍历输出
void ListPrint(LinkList L) {
//顺序输出单链表
LNode* p = L->next;
while (p) {
cout << p->data << "\t";
p = p->next;
}
cout << endl;
} int main() {
int i, n, e, choose;
LinkList L = InitList();
cout << "1. 初始化\n";
cout << "2. 创建单链表(头插法)\n";
cout << "3. 创建单链表(尾插法)\n";
cout << "4. 取值\n";
cout << "5. 查找\n";
cout << "6. 插入\n";
cout << "7. 删除\n";
cout << "8. 输出\n";
cout << "0. 退出\n";
choose = -1;
while (choose != 0) {
cout << "请输入数字进行操作: ";
cin >> choose;
switch (choose) {
case 1: //初始化一个空的单链表
L = InitList();
if (L) {
cout << "成功初始化一个空的单链表" << endl;
}else{
cout << "初始化一个空的单链表失败" << endl;
}
break;
case 2:// 创建单链表(头插法)
cout << "请输入想要创建链表元素个数:";
cin >> n;
cout << "请依次输入" << n << "个元素:";
CreateList_H(L, n);
cout << "头插法创建单链表输出结果:";
ListPrint(L);
break;
case 3:// 创建单链表(尾插法)
cout << "请输入想要创建链表元素个数:";
cin >> n;
cout << "请依次输入" << n << "个元素:";
CreateList_R(L, n);
cout << "尾插法创建单链表输出结果:";
ListPrint(L);
break;
case 4: //取值
cout << "请输入想查找的链表元素序号:";
cin >> i;
GetElem(L, i, e);
cout << "第" << i << "个链表元素的数据域是:" << e << endl;
break;
case 5: //查找
cout << "请输入想查找的链表元素的数据域:";
cin >> e;
if (LocateElem(L, e)) {
cout << "查找成功" << endl;
}
else {
cout << "查找失败" << endl;
}
break;
case 6: //插入
cout << "请输入插入的位置和数据(用空格隔开):";
cin >> i >> e;
if (InsertList(L, i, e)) {
cout << "插入成功" << endl;
}
else {
cout << "插入失败" << endl;
}
break;
case 7: //删除
cout << "请输入删除的位置:";
cin >> i;
if (DeleteList(L, i)) {
cout << "删除成功" << endl;
}
else {
cout << "删除失败" << endl;
}
break;
case 8: //遍历输出单链表
ListPrint(L);
break;
case 0: //退出
exit(-1);
}
}
return 0;
}

链表的应用

题目1:链表合并(Acwing3639)

有序链表的合并

题目:将两个有序(非递减)单链表La和Lb合并为一个新的有序(非递减)单链表。
解题思路:链表合并不需要再创建空间,只需要进行穿针引线,把两个单链表中的结点,按非递减的顺序串联起来即可。
注意:单链表的头指针不可以移动
# include <iostream>
using namespace std; typedef struct LNode {
int data; //结点的数据域
struct LNode* next; //结点的指针域
}LNode, * LinkList; //LinkList为指向结构体LNode的指针类型 //尾插法创建单链表
void CreateList_R(LinkList& L, int n) {
L = new LNode;
L->next = NULL; //创建一个空的头结点
LNode* r = L; //尾指针
for (int i = 0; i < n; i++) {
LNode* p = new LNode;
cin >> p->data;
r->next = p;
r = p;
}
r->next = NULL;//将最后一个结点的指针域置空
} LinkList MergeList(LinkList& L1, LinkList& L2) {
LNode* L = L1; //L是合并的新链表头指针,L指向L1的头结点
LNode* r = L; //尾指针永远指向链表尾部
LNode* p = L1->next, * q = L2->next;//p指针指向L1的第一个元素,q指针指向L2的第一个元素
while (p && q) { //p和q同时不指向空
if (p->data <= q->data) {//把p指向的结点串起来
r->next = p;
r = p;
p = p->next;//p后移到下一结点
}
else {//把q指向的结点串起来
r->next = q;
r = q;
q = q->next;
}
}
//退出循环的条件:1、p为空; 2、q为空
if (!p) r->next = q;//p为空
if (!q) r->next = p;//q为空
//或者r->next = p ? p : q;
delete L2;
return L;
}
void ListPrint(LinkList L) {
//顺序输出单链表
LNode* p = L->next;
while (p) {
cout << p->data << " ";
p = p->next;
}
cout << endl;
} int main() {
LinkList L1, L2, L;
int S1, S2;
cin >> S1;
CreateList_R(L1, S1);
cin >> S2;
CreateList_R(L2, S2);
L = MergeList(L1, L2);
ListPrint(L); return 0;
}
题目2: 链表的中间结点(Leetcode 872)

快慢指针

题目:带有头结点的单链表L,设计一个尽可能高效的算法,求L中的中间结点。如果有两个中间结点,则返回第一个中间结点
提示:给定链表的结点数介于 1 和 100 之间。
解题思路:这样的题型需要使用快慢指针来解决。一个快指针,一个慢指针,快指针走两步满指针走一步,当快指针指向结尾的时候,慢指针刚好指向中间结点
/*
题目:带有头结点的单链表L, 头结点的数据域不存储数据。设计一个尽可能高效的算法, 求L中的中间结点。
如果有两个中间结点, 则返回第一个中间结点.
*/
# include <iostream>
using namespace std; typedef struct LNode {
int data;
struct LNode* next;
}LNode, * LinkList; //尾插法建立单链表
void CreateList_R(LinkList& L, int n) {
L = new LNode;
L->next = NULL;
LNode* r = L; //尾指针
for (int i = 0; i < n; i++) {
LNode* p = new LNode;
cin >> p->data;
r->next = p;
r = p;
}
r->next = NULL; //链表最后一个结点的指针域置空
} LNode* FindMiddle(LinkList L) {
LNode* fast = L;
LNode* slow = L;
while (fast && fast->next) {
fast = fast->next->next; //快指针一次走两步
slow = slow->next; //慢指针一次走一步
}
return slow; //慢指针所指的位置就是中间结点的位置
} void PrintList(LinkList L) {
LNode* p = L->next;
while (p) {
cout << p->data << " ";
p = p->next;
}
cout << endl;
} int main() {
LinkList L;
LNode* middle;
int n;
cout << "输入链表结点个数:";
cin >> n;
cout << "依次输入" << n << "个元素:" ;
CreateList_R(L, n);
cout << "打印链表中的所有元素:";
PrintList(L);
middle = FindMiddle(L);
cout << "链表中间结点元素为:";
cout << middle->data << endl; system("pause");
return 0;
}
题目:
1.在单链表中查到倒数第k个结点
2.用单链表保存m个整数,结点的结构为(data,next),且|data|<=n(n为正整数)。现在要求设计一个时间复杂度尽可能高效的算法,对于链表中data的绝对值相等的结点,仅保留第一次出现的结点而阐述其绝对值相等的结点。

C++实现链表的相关基础操作的更多相关文章

  1. 【ADO.NET基础-GridView】GridView的编辑、更新、取消、删除以及相关基础操作代码

    代码都是基础操作,后续功能还会更新,如有问题欢迎提出和提问....... 前台代码: <asp:GridView ID=" OnRowDataBound="GridView1 ...

  2. [php入门] 3、WAMP中的集成MySQL相关基础操作

    前言:本文以小白视角了解WAMP集成开发环境中的MYSQL,涉及的面广而浅,算是导读性质. 1.启动运行熟悉WAMP中的MySQL 先有库.再有表.数据最终以记录的形式插入表中.其中对数据进行操作使用 ...

  3. git的相关基础操作

    一.git安装 从https://git-scm.com/下载相应版本安装即可,一路默认安装到底即可,安装目录可以自行选择 二.git配置 安装完git后在任意文件夹内单击鼠标右键,会出现Git GU ...

  4. python下selenium模拟浏览器基础操作

    1.安装及下载 selenium安装: pip install selenium  即可自动安装selenium geckodriver下载:https://github.com/mozilla/ge ...

  5. day06-Python运维开发基础(字符串格式化与相关的函数、列表相关的操作)

    1. 字符串相关的操作与格式化 # ### 字符串相关操作 # (1)字符串的拼接 + var1 = "亲爱的," var2 = "男孩" res = var1 ...

  6. NIO相关基础篇三

    转载请注明原创出处,谢谢! 说在前面 上篇NIO相关基础篇二,主要介绍了文件锁.以及比较关键的Selector,本篇继续NIO相关话题内容,主要谈谈一些Linux 网络 I/O模型.零拷贝等一些内容, ...

  7. python基础操作以及hdfs操作

    目录 前言 基础操作 hdfs操作 总结 一.前言        作为一个全栈工程师,必须要熟练掌握各种语言...HelloWorld.最近就被"逼着"走向了python开发之路, ...

  8. MYSQL基础操作

    MYSQL基础操作 [TOC] 1.基本定义 1.1.关系型数据库系统 关系型数据库系统是建立在关系模型上的数据库系统 什么是关系模型呢? 1.数据结构可以规定,同类数据结构一致,就是一个二维的表格 ...

  9. Emacs学习心得之 基础操作

    作者:枫雪庭 出处:http://www.cnblogs.com/FengXueTing-px/ 欢迎转载 Emacs学习心得之 基础操作 1.前言与学习计划2.Emacs基础操作 一. 前言与学习计 ...

随机推荐

  1. 4.QT:spinbox(spindoublebox)控件的信号响应

    Qt的QSpinBox和QDoubleSpinBox两个控件在默认情况下是valueChanged信号,会响应每次输入栏的改变. 比如想要输入数值"123",我们会依次键入1 - ...

  2. js jquey 笔记

    1.使用insertAfter 如果插入在tr后添加多行数据,数据会倒过来注意组字符串需要倒着循环 1).html方法,给元素添加html代码或者清空html代码(参数为空字符串): 2).appen ...

  3. Python迭代器和生成器你学会了吗?

    在了解什么是迭代器和生成器之前,我们先来了解一下容器的概念.对于一切皆对象来说,容器就是对象的集合.例如列表.元祖.字典等等都是容器.对于容器,你可以很直观地想象成多个元素在一起的单元:而不同容器的区 ...

  4. 资源:VMware秘钥许可证

    一. 激活密钥 YG5H2-ANZ0H-M8ERY-TXZZZ-YKRV8 UG5J2-0ME12-M89WY-NPWXX-WQH88 UA5DR-2ZD4H-089FY-6YQ5T-YPRX6 GA ...

  5. 1shell变量的作用域

    Shell 局部变量 Shell 全局变量 shell全局变量的易错点 linux shell中./a.sh , sh a.sh , source a.sh, . ./a.sh的区别 Shell 环境 ...

  6. Java实验项目三——递归实现字符串查找和替换操作

    Program:按照下面要求实现字符串的操作: (1)设计一个提供下面字符串操作的类 1)编写一个方法,查找在一个字符串中指定字符串出现的次数. 2)编写一个方法,参数(母字符串,目标字符串,替换字符 ...

  7. django 使用jpype 报错:raise+OSError('JVM+cannot+be+restarted')

    #调用jar包 def getJar(arg1,arg2): jarpath = os.path.join(os.path.abspath('.'), 'tools/GetTest-1.0-SNAPS ...

  8. postgresql 使用游标笔记

    游标介绍:游标是一种从表中检索数据并进行操作的灵活手段,游标主要用在服务器上,处理由客户端发送给服务端的sql语句,或是批处理.存储过程.触发器中的数据处理请求. 游标的优点在于它允许应用程序对查询语 ...

  9. 单点登录(SSO)实现原理(转)

    简介 单点登录是在多个应用系统中,用户只需要登录一次就可以访问所有相互信任的应用系统的保护资源,若用户在某个应用系统中进行注销登录,所有的应用系统都不能再直接访问保护资源,像一些知名的大型网站,如:淘 ...

  10. DHCP工作原理

    DHCP:Dynamic Host Configurtion Protocol DHCP的工作原理(UDP) 1.客户端:首先会发送给一个dhcp discovery(广播)报文,报文中的2层和3层都 ...