当前位置: 首页 > news >正文

链表中环的入口结点-C++

分享一个大牛的人工智能教程。零基础!通俗易懂!风趣幽默!希望你也加入到人工智能的队伍中来!请轻击人工智能教程https://www.captainai.net/troubleshooter

// 面试题23:链表中环的入口结点 // 题目:一个链表中包含环,如何找出环的入口结点?例如,在图3.8的链表中, // 环的入口结点是结点3。 #include <cstdio> #include <stdio.h> #include <stdlib.h> struct ListNode { int m_nValue; ListNode *m_pNext; }; ListNode *CreateListNode(int value) { ListNode *pNode = new ListNode(); pNode->m_nValue = value; pNode->m_pNext = nullptr; return pNode; } void ConnectListNodes(ListNode *pCurrent, ListNode *pNext) { if (pCurrent == nullptr) { printf("Error to connect two nodes.\n"); exit(1); } pCurrent->m_pNext = pNext; } void PrintListNode(ListNode *pNode) { if (pNode == nullptr) { printf("The node is nullptr\n"); } else { printf("The key in node is %d.\n", pNode->m_nValue); } } void PrintList(ListNode *pHead) { printf("PrintList starts.\n"); ListNode *pNode = pHead; while (pNode != nullptr) { printf("%d\t", pNode->m_nValue); pNode = pNode->m_pNext; } printf("\nPrintList ends.\n"); } void DestroyList(ListNode *pHead) { ListNode *pNode = pHead; while (pNode != nullptr) { pHead = pHead->m_pNext; delete pNode; pNode = pHead; } } void AddToTail(ListNode **pHead, int value) { ListNode *pNew = new ListNode(); pNew->m_nValue = value; pNew->m_pNext = nullptr; if (*pHead == nullptr) { *pHead = pNew; } else { ListNode *pNode = *pHead; while (pNode->m_pNext != nullptr) pNode = pNode->m_pNext; pNode->m_pNext = pNew; } } void RemoveNode(ListNode **pHead, int value) { if (pHead == nullptr || *pHead == nullptr) return; ListNode *pToBeDeleted = nullptr; if ((*pHead)->m_nValue == value) { pToBeDeleted = *pHead; *pHead = (*pHead)->m_pNext; } else { ListNode *pNode = *pHead; while (pNode->m_pNext != nullptr && pNode->m_pNext->m_nValue != value) pNode = pNode->m_pNext; if (pNode->m_pNext != nullptr && pNode->m_pNext->m_nValue == value) { pToBeDeleted = pNode->m_pNext; pNode->m_pNext = pNode->m_pNext->m_pNext; } } if (pToBeDeleted != nullptr) { delete pToBeDeleted; pToBeDeleted = nullptr; } } ListNode *MeetingNode(ListNode *pHead) { if (pHead == nullptr) return nullptr; ListNode *pSlow = pHead->m_pNext; if (pSlow == nullptr) return nullptr; ListNode *pFast = pSlow->m_pNext; while (pFast != nullptr && pSlow != nullptr) { if (pFast == pSlow) return pFast; pSlow = pSlow->m_pNext; pFast = pFast->m_pNext; if (pFast != nullptr) pFast = pFast->m_pNext; } return nullptr; } ListNode *EntryNodeOfLoop(ListNode *pHead) { ListNode *meetingNode = MeetingNode(pHead); if (meetingNode == nullptr) return nullptr; // 得到环中结点的数目 int nodesInLoop = 1; ListNode *pNode1 = meetingNode; while (pNode1->m_pNext != meetingNode) { pNode1 = pNode1->m_pNext; ++nodesInLoop; } // 先移动pNode1,次数为环中结点的数目 pNode1 = pHead; for (int i = 0; i < nodesInLoop; ++i) pNode1 = pNode1->m_pNext; // 再移动pNode1和pNode2 ListNode *pNode2 = pHead; while (pNode1 != pNode2) { pNode1 = pNode1->m_pNext; pNode2 = pNode2->m_pNext; } return pNode1; } // ==================== Test Code ==================== void Test(char *testName, ListNode *pHead, ListNode *entryNode) { if (testName != nullptr) printf("%s begins: ", testName); if (EntryNodeOfLoop(pHead) == entryNode) printf("Passed.\n"); else printf("FAILED.\n"); } // A list has a node, without a loop void Test1() { ListNode *pNode1 = CreateListNode(1); Test("Test1", pNode1, nullptr); DestroyList(pNode1); } // A list has a node, with a loop void Test2() { ListNode *pNode1 = CreateListNode(1); ConnectListNodes(pNode1, pNode1); Test("Test2", pNode1, pNode1); delete pNode1; pNode1 = nullptr; } // A list has multiple nodes, with a loop void Test3() { ListNode *pNode1 = CreateListNode(1); ListNode *pNode2 = CreateListNode(2); ListNode *pNode3 = CreateListNode(3); ListNode *pNode4 = CreateListNode(4); ListNode *pNode5 = CreateListNode(5); ConnectListNodes(pNode1, pNode2); ConnectListNodes(pNode2, pNode3); ConnectListNodes(pNode3, pNode4); ConnectListNodes(pNode4, pNode5); ConnectListNodes(pNode5, pNode3); Test("Test3", pNode1, pNode3); delete pNode1; pNode1 = nullptr; delete pNode2; pNode2 = nullptr; delete pNode3; pNode3 = nullptr; delete pNode4; pNode4 = nullptr; delete pNode5; pNode5 = nullptr; } // A list has multiple nodes, with a loop void Test4() { ListNode *pNode1 = CreateListNode(1); ListNode *pNode2 = CreateListNode(2); ListNode *pNode3 = CreateListNode(3); ListNode *pNode4 = CreateListNode(4); ListNode *pNode5 = CreateListNode(5); ConnectListNodes(pNode1, pNode2); ConnectListNodes(pNode2, pNode3); ConnectListNodes(pNode3, pNode4); ConnectListNodes(pNode4, pNode5); ConnectListNodes(pNode5, pNode1); Test("Test4", pNode1, pNode1); delete pNode1; pNode1 = nullptr; delete pNode2; pNode2 = nullptr; delete pNode3; pNode3 = nullptr; delete pNode4; pNode4 = nullptr; delete pNode5; pNode5 = nullptr; } // A list has multiple nodes, with a loop void Test5() { ListNode *pNode1 = CreateListNode(1); ListNode *pNode2 = CreateListNode(2); ListNode *pNode3 = CreateListNode(3); ListNode *pNode4 = CreateListNode(4); ListNode *pNode5 = CreateListNode(5); ConnectListNodes(pNode1, pNode2); ConnectListNodes(pNode2, pNode3); ConnectListNodes(pNode3, pNode4); ConnectListNodes(pNode4, pNode5); ConnectListNodes(pNode5, pNode5); Test("Test5", pNode1, pNode5); delete pNode1; pNode1 = nullptr; delete pNode2; pNode2 = nullptr; delete pNode3; pNode3 = nullptr; delete pNode4; pNode4 = nullptr; delete pNode5; pNode5 = nullptr; } // A list has multiple nodes, without a loop void Test6() { ListNode *pNode1 = CreateListNode(1); ListNode *pNode2 = CreateListNode(2); ListNode *pNode3 = CreateListNode(3); ListNode *pNode4 = CreateListNode(4); ListNode *pNode5 = CreateListNode(5); ConnectListNodes(pNode1, pNode2); ConnectListNodes(pNode2, pNode3); ConnectListNodes(pNode3, pNode4); ConnectListNodes(pNode4, pNode5); Test("Test6", pNode1, nullptr); DestroyList(pNode1); } // Empty list void Test7() { Test("Test7", nullptr, nullptr); } int main(int argc, char *argv[]) { Test1(); Test2(); Test3(); Test4(); Test5(); Test6(); Test7(); return 0; } // ------ Output ------ /* Test1 begins: Passed. Test2 begins: Passed. Test3 begins: Passed. Test4 begins: Passed. Test5 begins: Passed. Test6 begins: Passed. Test7 begins: Passed. */
http://www.jsqmd.com/news/712139/

相关文章:

  • 2026年3月高效的宠物医院运营托管团队推荐,宠物医院代运营/宠物医生美团运营,宠物医院运营托管品牌怎么选择 - 品牌推荐师
  • 如何利用Turborepo实现TypeScript项目的类型安全构建流程优化
  • 多项式优化与半定规划松弛的计算挑战与优化策略
  • 红外线桥切机哪家好?桥切机厂家有哪些?2026年桥切机厂家推荐:福建晶洋领衔 - 栗子测评
  • 2026乐山油炸工艺解析:乐山美食攻略、乐山美食街、乐山美食订餐热线、乐山辜李坝老地方油炸、乐山市区美食、乐山当地人去的美食街选择指南 - 优质品牌商家
  • 深度解析AssetStudio:从Unity资源提取到Lua字节码反编译的完整解决方案
  • Python 上下文管理器:高级应用
  • YOLOv8搭配5大跟踪算法实测对比:DeepOCSort、StrongSORT、OCSort、ByteTrack、BoT-SORT哪个更适合你的项目?
  • 涡旋压缩机设计(说明书+CAD图纸+UG三维模型+开题报告+实习报告+答辩PPT+外文翻译+文献综述)
  • AI论文精华速递:三重过滤机制与关键技术解析
  • AMD EPYC 9005嵌入式处理器:Zen 5架构与CXL 2.0技术解析
  • Android开发技术选型终极指南:框架、库与工具的综合评估
  • 如何用AI驱动组件库彻底改变前端开发:GitHub_Trending/ui/ui的终极指南
  • 2026年筛网围栏生产厂家/不锈钢筛网源头厂家推荐:洲冠领衔,优质316不锈钢筛网生产厂商/304不锈钢筛网生产厂家盘点 - 栗子测评
  • PaperClaw:为科研团队构建AI驱动的知识协作与合成工作流
  • 小型语言模型在金融价格预测中的高效实践
  • XState撤销重做:用户操作历史管理的终极实现指南
  • TestDisk PhotoRec:开源数据恢复双雄,从分区修复到文件拯救的完整指南
  • ARM GIC中断控制器虚拟化与EL2陷阱机制详解
  • 反转链表-C++
  • 浅谈现代物流中的自动化立体仓库毕业设计
  • VFP JSON处理利器nfJson:纯代码实现、高性能解析与实战应用
  • TypeScript Go终极指南:如何快速掌握TypeScript原生移植技术
  • docker-compose安装
  • 彻底解决Prisma事务超时:Node进程崩溃的终极指南
  • 深度学习优化:学习率调度与早停
  • 从‘乱码’到‘清晰’:深入理解JavaScript中Base64编码的字符集‘暗礁’与安全实践
  • 告别组件绑定困境:Dapr插件架构如何重塑云原生扩展能力
  • 2026液压家用电梯技术分享:山东别墅电梯、山东家用电梯、螺杆电梯、观光电梯、三层电梯、二层电梯、室内电梯、室外电梯选择指南 - 优质品牌商家
  • JCSprout算法优化:空间换时间策略的终极指南