从尾到头打印单向链表

2023-12-02 11:32

本文主要是介绍从尾到头打印单向链表,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1.需求及分析

输入一个链表的头结点,从尾到头反过来打印出每个节点的值。

方法:

1.把链表中链接节点的指针反转过来,改变链表的方向,然后从头到尾输出。(实际上修改了链表的结构下下策);

2.典型的先进后出,可以用“栈”处理;

3.典型的先进后出,由于递归和栈的处理方式接近也可以用递归处理。

2.使用栈处理

void PrintListReversingly_Iter(ListNode* pHead)
{std::stack<ListNode*> nodes; //定义链表指针栈区ListNode* pNode = pHead;     // 链表首地址指针while (pNode != NULL) //终止条件:链表地址指针为空{nodes.push(pNode);       //链表首地址指针压入栈区pNode = pNode->next;     //取下一个指针}while (!nodes.empty()) //终止条件:栈区为空{pNode = nodes.top();             //弹出链表指针cout << pNode->value<<"       "; //取出索引值nodes.pop();}
}

3.等效递归处理

void PrintListReversingly_Recur(ListNode* pHead)
{if (pHead != NULL){if (pHead->next != NULL){PrintListReversingly_Recur(pHead->next);}cout << pHead->value << "       "; }
}

4.测试代码

/* list.h */
#include <stdio.h>  
#include <stdlib.h>  struct ListNode{int value;ListNode* next;
};ListNode* CreateListNode(int value){ListNode* pNode = new ListNode();pNode->value = value;pNode->next = NULL;return pNode;
}void ConnectListNode(ListNode* pCurrent, ListNode* pNext){if (pCurrent == NULL){printf("Error to connect two nodes.\n");exit(1);}pCurrent->next = pNext;
}void PrintListNode(ListNode* pNode){if (pNode == NULL)printf("The node is null.\n");elseprintf("The key in node is %d.\n", pNode->value);
}void PrintList(ListNode* pHead){ListNode* pNode = pHead;while (pNode){printf("%d\t", pNode->value);pNode = pNode->next;}printf("\n");
}void DestroyList(ListNode* pHead){ListNode* pNode = pHead;while (pNode){ListNode* pNext = pNode->next;delete pNode;pNode = pNext;}
}void AddToTail(ListNode** pHead, int value){ListNode* pNode = new ListNode();pNode->value = value;pNode->next = NULL;if (*pHead == NULL)*pHead = pNode;else{ListNode* pCurrent = *pHead;while (pCurrent->next)pCurrent = pCurrent->next;pCurrent->next = pNode;}
}void RemoveHead(ListNode** pHead, int value){if (pHead == NULL || *pHead == NULL)return;ListNode* pToBeDeleted = NULL;if ((*pHead)->value == value){pToBeDeleted = *pHead;*pHead = (*pHead)->next;}else{ListNode* pNode = *pHead;while (pNode->next != NULL && pNode->next->value != value)pNode = pNode->next;if (pNode->next != NULL && pNode->next->value == value){pToBeDeleted = pNode->next;pNode->next = pNode->next->next;}}if (pToBeDeleted != NULL){delete pToBeDeleted;pToBeDeleted = NULL;}
}
/* PrintReversingly */
#include <iostream>
#include "list.h"
#include <stack>
using namespace std;void PrintListReversingly_Iter(ListNode* pHead)
{std::stack<ListNode*> nodes; //定义链表指针栈区ListNode* pNode = pHead;     // 链表首地址指针while (pNode != NULL) //终止条件:链表地址指针为空{nodes.push(pNode);       //链表首地址指针压入栈区pNode = pNode->next;     //取下一个指针}while (!nodes.empty()) //终止条件:栈区为空{pNode = nodes.top();             //弹出链表指针cout << pNode->value<<"       "; //取出索引值nodes.pop();}
}void PrintListReversingly_Recur(ListNode* pHead)
{if (pHead != NULL){if (pHead->next != NULL){PrintListReversingly_Recur(pHead->next);}cout << pHead->value << "       "; }
}
void Results(ListNode* pHead)
{PrintList(pHead);cout << endl;PrintListReversingly_Iter(pHead);cout << endl;PrintListReversingly_Recur(pHead);
}void Test1()
{cout << endl;cout << "Test1 begins:" << endl;ListNode* pNode1 = CreateListNode(1);ListNode* pNode2 = CreateListNode(2);ListNode* pNode3 = CreateListNode(3);ListNode* pNode4 = CreateListNode(4);ListNode* pNode5 = CreateListNode(5);ConnectListNode(pNode1, pNode2);ConnectListNode(pNode2, pNode3);ConnectListNode(pNode3, pNode4);ConnectListNode(pNode4, pNode5);Results(pNode1);DestroyList(pNode1);
}int main()
{Test1();return 0;
}


这篇关于从尾到头打印单向链表的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/445187

相关文章

csu1329(双向链表)

题意:给n个盒子,编号为1到n,四个操作:1、将x盒子移到y的左边;2、将x盒子移到y的右边;3、交换x和y盒子的位置;4、将所有的盒子反过来放。 思路分析:用双向链表解决。每个操作的时间复杂度为O(1),用数组来模拟链表,下面的代码是参考刘老师的标程写的。 代码如下: #include<iostream>#include<algorithm>#include<stdio.h>#

深入手撕链表

链表 分类概念单链表增尾插头插插入 删尾删头删删除 查完整实现带头不带头 双向链表初始化增尾插头插插入 删查完整代码 数组 分类 #mermaid-svg-qKD178fTiiaYeKjl {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-

建立升序链表

题目1181:遍历链表 时间限制:1 秒 内存限制:32 兆 特殊判题:否 提交:2744 解决:1186 题目描述: 建立一个升序链表并遍历输出。 输入: 输入的每个案例中第一行包括1个整数:n(1<=n<=1000),接下来的一行包括n个整数。 输出: 可能有多组测试数据,对于每组数据, 将n个整数建立升序链表,之后遍历链表并输出。 样例输

【每日一题】LeetCode 2181.合并零之间的节点(链表、模拟)

【每日一题】LeetCode 2181.合并零之间的节点(链表、模拟) 题目描述 给定一个链表,链表中的每个节点代表一个整数。链表中的整数由 0 分隔开,表示不同的区间。链表的开始和结束节点的值都为 0。任务是将每两个相邻的 0 之间的所有节点合并成一个节点,新节点的值为原区间内所有节点值的和。合并后,需要移除所有的 0,并返回修改后的链表头节点。 思路分析 初始化:创建一个虚拟头节点

学习记录:js算法(二十八):删除排序链表中的重复元素、删除排序链表中的重复元素II

文章目录 删除排序链表中的重复元素我的思路解法一:循环解法二:递归 网上思路 删除排序链表中的重复元素 II我的思路网上思路 总结 删除排序链表中的重复元素 给定一个已排序的链表的头 head , 删除所有重复的元素,使每个元素只出现一次 。返回 已排序的链表 。 图一 图二 示例 1:(图一)输入:head = [1,1,2]输出:[1,2]示例 2:(图

多数据源的事务处理总是打印很多无用的log日志

之前做了一个项目,需要用到多数据源以及事务处理,在使用事务处理,服务器总是打印很多关于事务处理的log日志(com.atomikos.logging.Slf4jLogger),但是我们根本不会用到这些log日志,反而使得查询一些有用的log日志变得困难。那要如何屏蔽这些log日志呢? 之前的项目是提高项目打印log日志的级别,后来觉得这样治标不治本。 现在有一个更好的方法: 我使用的是log

fastreport打印trichedit分页问题的解决

用fastreport来打印richedit里面的内容。刚开始放一个frxrichview组件到报表上,然后在 var str: TMemoryStream; begin    begin      str:= TMemoryStream.Create;      CurrRichRecord.richedit.Lines.SaveToStream(str);      str.Posit

模具要不要建设3D打印中心

随着3D打印技术的日益成熟与广泛应用,模具企业迎来了自建3D打印中心的热潮。这一举措不仅为企业带来了前所未有的发展机遇,同时也伴随着一系列需要克服的挑战,如何看待企业引进增材制造,小编为您全面分析。 机遇篇: 加速产品创新:3D打印技术如同一把钥匙,为模具企业解锁了快速迭代产品设计的可能。企业能够迅速将创意转化为实体模型,缩短产品从设计到市场的周期,抢占市场先机。 强化定制化服务:面

【数据结构与算法 | 灵神题单 | 删除链表篇】力扣3217, 82, 237

总结,删除链表节点问题使用到列表,哈希表,递归比较容易超时,我觉得使用计数排序比较稳,处理起来也不是很难。 1. 力扣3217:从链表中移除在数组中的节点 1.1 题目: 给你一个整数数组 nums 和一个链表的头节点 head。从链表中移除所有存在于 nums 中的节点后,返回修改后的链表的头节点。 示例 1: 输入: nums = [1,2,3], head = [1,2,3,

c++ 链表详细介绍

链表是数据结构的一种,由节点组成,每个节点包含数据和指向下一个节点的指针。链表在C++中的实现可以是单链表、双链表或循环链表。以下是链表的详细介绍: 1. 单链表 结构: 节点(Node):每个节点包含数据和一个指针(next),指向链表中的下一个节点。 示例结构: struct Node {int data;Node* next;Node(int d) : data(d), next(