C/C++每日一练:查找链表的中间节点

链表(Linked List)

         链表是一种线性数据结构,由一系列节点(Node)通过指针链接在一起。与数组不同,链表中的元素在内存中不需要连续存储,每个节点包含两部分:

  • 数据部分:存储节点的值或数据。
  • 指针部分:存储指向下一个节点的地址(单链表)或上一个和下一个节点的地址(双向链表)。

         链表的类型主要有以下几种:

  • 单链表:每个节点只指向下一个节点。
  • 双向链表:每个节点既有指向下一个节点的指针,也有指向上一个节点的指针。
  • 循环链表:链表的最后一个节点指向链表的头节点,形成循环。

         链表的特点:

  • 动态大小:可以根据需要动态地增加或减少元素,无需预分配存储空间。
  • 插入/删除效率高:在链表的任意位置进行插入或删除操作只需修改指针,不涉及大量元素的移动,效率较高。
  • 随机访问效率低:由于链表不支持直接访问任意位置的元素,需要通过遍历来查找特定位置的节点。

         如下图所示:

题目要求

         给定一个单链表,编写一个算法来查找链表的中间节点。如果链表长度为奇数,则返回唯一的中间节点;如果为偶数,则返回两个中间节点中的第一个。

示例

假设链表为:1 -> 2 -> 3 -> 4 -> 5

  • 输出:节点 3

假设链表为:1 -> 2 -> 3 -> 4

  • 输出:节点 2

做题思路

         寻找链表的中间节点可以用快慢指针的方法。这种方法效率高且不需要额外的空间,适合单链表查找问题。

  1. 定义两个指针:slow 和 fast。
  2. slow 每次向前移动一步,fast 每次向前移动两步。
  3. 当 fast 指针遍历到达链表末尾时,slow 指针正好在链表的中间位置。
  4. 这种方法的时间复杂度为 O(n),空间复杂度为 O(1),是一种高效的解法。

过程解析

  1. 初始化两个指针 slow 和 fast,都指向链表的头节点。
  2. 通过循环控制 fast 每次移动两步,slow 每次移动一步。
  3. 当 fast 为 NULL 或 fast->next 为 NULL 时,结束循环,此时 slow 即为中间节点。

运用到的知识点

  • 链表的基本操作和定义
  • 指针的概念和指针操作
  • 快慢指针算法技巧
  • 时间和空间复杂度分析

示例代码

C 实现

#include <stdio.h>   // 包含标准输入输出库,用于输出打印函数
#include <stdlib.h>  // 包含标准库,用于内存分配// 定义链表节点结构体
struct Node {int data;            // 节点数据部分,存储整型数据struct Node* next;   // 指向下一个节点的指针
};// 创建新节点函数
struct Node* createNode(int data) {// 为新节点分配内存空间struct Node* newNode = (struct Node*)malloc(sizeof(struct Node));// 初始化新节点的数据newNode->data = data;// 将新节点的指针设置为 NULL,表示链表的末尾newNode->next = NULL;return newNode;  // 返回创建好的新节点
}// 查找链表中间节点的函数
struct Node* findMiddle(struct Node* head) {struct Node *slow = head, *fast = head;  // 定义并初始化快慢指针,均指向链表头节点// 循环遍历链表,直到 fast 到达链表末尾while (fast != NULL && fast->next != NULL) {slow = slow->next;        // 慢指针每次向前移动一步fast = fast->next->next;  // 快指针每次向前移动两步}return slow;  // 当循环结束时,慢指针指向链表的中间节点
}// 打印链表的函数
void printList(struct Node* head) {struct Node* temp = head;  // 定义一个临时指针,用于遍历链表// 循环遍历链表,直到节点为空while (temp != NULL) {printf("%d -> ", temp->data);  // 打印当前节点的数据temp = temp->next;             // 移动到下一个节点}printf("NULL\n");  // 链表末尾指示为 NULL
}// 主函数
int main() 
{// 创建链表节点并连接,构造链表 1 -> 2 -> 3 -> 4 -> 5struct Node* head = createNode(1);head->next = createNode(2);head->next->next = createNode(3);head->next->next->next = createNode(4);head->next->next->next->next = createNode(5);printf("链表: ");printList(head);  // 调用函数打印链表// 查找链表的中间节点struct Node* middle = findMiddle(head);if (middle != NULL) {// 如果链表非空,打印中间节点的数据printf("The data of the middle node is: %d\n", middle->data);} else {// 如果链表为空,提示信息printf("the linked list is empty.\n");}return 0;  // 程序结束
}

C++ 实现

#include <iostream>   // 包含输入输出流库,用于控制台输出
using namespace std;  // 使用标准命名空间,避免每次使用 std:: 前缀// 定义链表节点类
class Node {
public:int data;      // 节点数据部分,存储整型数据Node* next;    // 指向下一个节点的指针// 构造函数,用于创建新节点并初始化数据Node(int data) {this->data = data;      // 初始化节点数据this->next = nullptr;   // 初始化指针为 nullptr 表示链表末尾}
};// 查找链表中间节点的函数
Node* findMiddle(Node* head) {Node *slow = head, *fast = head;  // 定义快慢指针,均初始化为链表头节点// 循环遍历链表,直到 fast 或 fast->next 到达链表末尾while (fast != nullptr && fast->next != nullptr) {slow = slow->next;        // 慢指针每次移动一步fast = fast->next->next;  // 快指针每次移动两步}return slow;  // 循环结束时,慢指针指向链表的中间节点
}// 打印链表的函数
void printList(Node* head) {Node* temp = head;  // 定义临时指针,用于遍历链表// 循环遍历链表,直到节点为空while (temp != nullptr) {cout << temp->data << " -> ";  // 打印当前节点的数据temp = temp->next;             // 移动到下一个节点}cout << "NULL" << endl;  // 链表末尾输出 NULL 表示链表终点
}int main() 
{// 创建链表节点并连接,构造链表 1 -> 2 -> 3 -> 4 -> 5Node* head = new Node(1);head->next = new Node(2);head->next->next = new Node(3);head->next->next->next = new Node(4);head->next->next->next->next = new Node(5);cout << "链表: ";printList(head);  // 打印链表// 查找链表的中间节点Node* middle = findMiddle(head);if (middle != nullptr) {// 如果链表非空,打印中间节点的数据cout << "The data of the intermediate node is: " << middle->data << endl;} else {// 如果链表为空,提示信息cout << "The linked list is empty" << endl;}return 0;  // 程序结束
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/diannao/60471.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

对称加密与非对称加密:密码学的基石及 RSA 算法详解

对称加密与非对称加密&#xff1a;密码学的基石及 RSA 算法详解 在当今数字化的时代&#xff0c;信息安全至关重要。对称加密和非对称加密作为密码学中的两种基本加密技术&#xff0c;为我们的数据安全提供了强大的保障。本文将深入探讨对称加密和非对称加密的特点、应用场景&…

PH47代码框架全局函数及功能类

PH47代码框架全局函数及功能类 概述 全局函数及功能类体系是PH47框架当中除了4个逻辑层之外最重要的组成部分之一&#xff0c;它们可以在 整个PH7 代码框架及用户代码中使用。常用全局函数及功能类为 PH7 代码框架提供了最常用和最基础的功能实现。 全局函数主要包含了对时间…

力扣 LeetCode 203. 移除链表元素(Day2:链表)

解题思路&#xff1a; 方法一&#xff1a;头节点和非头节点分开处理 方法二&#xff1a;定义一个dummy虚拟节点&#xff0c;后面的节点就可以采用相同的处理方式 注意&#xff1a; cur需要指向要删除的节点的上一个节点&#xff0c;因为要越过这一个被删除的节点 class Sol…

IEC60870-5-104 协议源码架构详细分析

IEC60870-5-104 协议源码架构 前言一、资源三、目录层级一二、目录层级二config/lib60870_config.hdependencies/READMEexamplesCMakeLists.txtcs101_master_balancedcs104_client_asyncmulti_client_servertls_clienttls_server说明 make这些文件的作用是否需要导入这些文件&a…

turbo c 2.0 画螺栓

代码; #include<graphics.h> void bolt(x0,y0,d,l) int x0,y0,d,l; {int x1,x2,x3,x4,x5,x6,x7,x8;int y1,y2,y3,y4,y5,r1,r2,b,c;if(l>2*d) b2*d;else b1;r11.5*d;r20.38*d;c0.1*d;x1x0-0.7*d;x2x0-0.61*d;x3x0-0.32*d;x4x00.8*d;x5x0l-b;x6x0l-c;x7x0l-0.05*d;x8x0…

网络服务综合项目-博客

一、运行环境&#xff1a; 主机主机名系统服务192.168.31.128Server-WebLinuxWeb192.168.31.129Server-NFS-DNSLinuxNFS 二、基础配置&#xff1a; 配置主机名开启防火墙并配置部分开启selinux并配置服务器之间使用ntp.aliyun.com进行时间同步服务器之间实现ssh免密登录 三…

leetcode86:分隔链表

给你一个链表的头节点 head 和一个特定值 x &#xff0c;请你对链表进行分隔&#xff0c;使得所有 小于 x 的节点都出现在 大于或等于 x 的节点之前。 你应当 保留 两个分区中每个节点的初始相对位置。 示例 1&#xff1a; 输入&#xff1a;head [1,4,3,2,5,2], x 3 输出&am…

Android Mobile Network Settings | APN 菜单加载异常

问题 从log看是有创建APN对应的Controller&#xff08;功能逻辑是ok的&#xff09;&#xff0c;但是Mobile Network Settings无法显示&#xff08;UI异常&#xff09;。 日志分析 看似APN 菜单已经创建了&#xff0c;实际上并没有显示。 11-12 07:01:28.150 8773 8773 D Pr…

上海市计算机学会竞赛平台2020年4月月赛丙组永恒的生命游戏

题目背景 2020年4月11日&#xff0c;英国数学家 约翰霍顿康威&#xff08;John Horton Conway&#xff09;因为新型冠状病毒肺炎不幸逝世。他在群论、数论、代数、几何拓扑、理论物理、组合博弈论和几何等领域&#xff0c;都做出了重大贡献。他的离去是人类文明的损失。他最著…

FS8x 功能安全

fail-safe是电独立的和物理隔离的。fail-safe由自己的参考电压和电流提供,有自己的振荡器,有重复的模拟路径以最小化常见的故障,并有LBIST/ABIST来覆盖潜在故障。fail-safe根据设备部件号提供ASIL B或ASIL D遵从性。除非另有规定,fail-safe定时来自故障安全振荡器,其精度为…

项目模块十七:HttpServer模块

一、项目模块设计思路 目的&#xff1a;实现HTTP服务器搭建 思想&#xff1a;设计请求路由表&#xff0c;记录请求方法与对应业务的处理函数映射关系。用户实现请求方法和处理函数添加到路由表&#xff0c;服务器只接受请求并调用用户的处理函数即可。 处理流程&#xff1a; …

内网域环境、工作组、局域网等探针方案

1. 信息收集 1.1 网络收集 了解当前服务器的计算机基本信息&#xff0c;为后续判断服务器角色&#xff0c;网络环境做准备 systeminfo 详细信息 net start 启动服务 tasklist 进程列表 schtasks 计划任务&#xff08;受权限影响&#xff09; 了解当前服务器的网络接口信息…

什么是量化交易

课程大纲 内容初级初识量化&#xff0c;理解量化 初识量化 传统量化和AI量化的区别 量化思想挖掘 量化思想的挖掘及积累技巧 量化代码基础&#xff1a; python、pandas、SQL基础语法 金融数据分析 常用金融分析方式 常用因子分析方式 数据分析实战练习 回测及交易引擎 交易引擎…

OpenHarmony-1.启动流程

OpenHarmony启动流程 1.kernel的启动 流程图如下所示&#xff1a;   OpenHarmony(简称OH)的标准系统的底层系统是linux&#xff0c;所以调用如下代码&#xff1a; linux-5.10/init/main.c: noinline void __ref rest_init(void) {struct task_struct *tsk;int pid;rcu_sch…

【LeetCode】【算法】64. 最小路径和

LeetCode 64. 最小路径和 题目描述 给定一个包含非负整数的 m x n 网格 grid &#xff0c;请找出一条从左上角到右下角的路径&#xff0c;使得路径上的数字总和为最小。 说明&#xff1a;每次只能向下或者向右移动一步。 思路 思路&#xff1a;这种题太典了&#xff0c;典…

1.7 JS性能优化

从输入url到页面加载完成都做了些什么 输入 URL - 资源定位符 http://www.zhaowa.com - http 协议 追问&#xff1a;http 与 TCP 1. http - 应用层 < > TCP - 传输层 2. 关联 - http基于TCP实现连接 < > UDP > 握手 & 挥手 &#xff08;传输速率上较…

Spring Task详细讲解

✨Spring Task简介 Spring Task 是 Spring 提供的轻量级定时任务工具&#xff0c;也就意味着不需要再添加第三方依赖了&#xff0c;相比其他第三方类库更加方便易用。可以按照约定的时间自动执行某个代码逻辑。 使用场景&#xff1a; 信用卡每月还款提醒银行贷款每月还款提醒…

Qt/C++ 海康SDK开发示例Demo

*** 工业相机在机器视觉中起到关键作用&#xff0c;本文基于海康 SDK 详细解读了设备连接与控制的各个步骤。内容涵盖设备枚举、句柄创建、图像采集回调以及设备异常处理&#xff0c;帮助开发者快速理解如何通过代码控制相机&#xff0c;实时采集并处理图像数据。*** 1. 搜索并…

HDLBIts习题(5):移位寄存器

&#xff08;1&#xff09;易错习题1&#xff1a;109题&#xff08;shift18&#xff09; 对算数左移和算数右移概念不清&#xff0c;不知道该如何计算。 逻辑左移和算术左移之间没有区别。&#xff08;无论是有符号位数据还是无符号位数据&#xff0c;右侧补0&#xff09; 逻辑…

想要成为独立游戏作者 :通关!游戏设计之道 2-2 关卡设计

本文通过ai辅助总结加个人微调,不喜勿喷 前篇如下&#xff1a; 想要成为独立游戏作者 &#xff1a;通关&#xff01;游戏设计之道 2-1 HUD-CSDN博客 1.关卡的多重定义 在电子游戏行业里 “关卡” 有多种含义&#xff0c;如游戏行为发生的环境、分割的游戏体验单元、量…