数据结构练习题4(链表)

news2025/1/17 6:10:36

1两两交换链表中的节点

给你一个链表,两两交换其中相邻的节点,并返回交换后链表的头节点。你必须在不修改节点内部的值的情况下完成本题(即,只能进行节点交换)。

示例 1:

输入:head = [1,2,3,4]
输出:[2,1,4,3]

示例 2:

输入:head = []
输出:[]

示例 3:

输入:head = [1]
输出:[1]

提示:

  • 链表中节点的数目在范围 [0, 100] 内
  • 0 <= Node.val <= 100

思路:

初始检查:如果链表为空或只有一个节点,直接返回。
创建虚拟头节点:使用一个虚拟头节点 dummy 来简化边界情况的处理。
循环交换节点对:在链表中继续交换每一对节点,直到没有更多的节点对(cur->next && cur->next->next 都不为空时)可以交换。开始时cur为与虚拟头结点位置
步骤一:cur->next 指向第二个节点。
步骤二:第二个节点的 next 指向第一个节点。
步骤三:第一个节点的 next 指向第三个节点。
移动到下一对节点的前一个节点:将 cur 移动到下一对节点的前一个节点。

代码:

struct ListNode* swapPairs(struct ListNode* head) {
    // 如果链表为空或只有一个节点,直接返回
    if (!head || !head->next) return head;

    struct ListNode dummy;  // 创建一个虚拟头节点
    dummy.next = head;
    struct ListNode* cur = &dummy;

    while (cur->next && cur->next->next) {
        struct ListNode* tmp = cur->next;         // 保存第一个节点
        struct ListNode* tmp1 = cur->next->next->next;  // 保存第二个节点的下一个节点

        cur->next = cur->next->next;    // 步骤一:cur->next 指向第二个节点
        cur->next->next = tmp;          // 步骤二:第二个节点的 next 指向第一个节点
        cur->next->next->next = tmp1;   // 步骤三:第一个节点的 next 指向第三个节点

        cur = cur->next->next;  // 移动到下一对节点的前一个节点
    }

    return dummy.next;  // 返回新的头节点
}

2删除链表的倒数第 N 个结点

给你一个链表,删除链表的倒数第 n 个结点,并且返回链表的头结点。

示例 1:

输入:head = [1,2,3,4,5], n = 2
输出:[1,2,3,5]

示例 2:

输入:head = [1], n = 1
输出:[]

示例 3:

输入:head = [1,2], n = 1
输出:[1]

提示:

  • 链表中结点的数目为 sz
  • 1 <= sz <= 30
  • 0 <= Node.val <= 100
  • 1 <= n <= sz

思路:

  1. 创建虚拟头节点:使用 malloc 分配内存并创建一个虚拟头节点 dummy,其 val 为 0,next 指向实际的头节点 head
  2. 初始化快慢指针fast 和 slow 都指向虚拟头节点。
  3. 快指针先移动 n 步:通过循环,让 fast 指针先移动 n 步。
  4. 快指针再移动一步:因为需要让slow指向删除节点的上一个节点
  5. 同时移动快慢指针:继续移动 fast 和 slow 指针,直到 fast 指针到达链表末尾。此时,slow 指针指向的节点就是要删除的节点的前一个节点。
  6. 删除节点:将 slow 的 next 指向 slow->next->next,从而删除目标节点。
  7. 更新头节点:将 head 更新为 dummy->next
  8. 释放虚拟头节点:使用 free 释放虚拟头节点的内存空间。
  9. 返回头节点:返回更新后的头节点 head

代码:

/**
 * Definition for singly-linked list.
 * struct ListNode {
 *     int val;
 *     struct ListNode *next;
 * };
 */
struct ListNode* removeNthFromEnd(struct ListNode* head, int n) {
    // 创建一个虚拟头节点,并初始化其值为0,next指向实际的头节点
    struct ListNode* dummy = malloc(sizeof(struct ListNode));
    dummy->val = 0;
    dummy->next = head;

    // 初始化快慢指针,都指向虚拟头节点
    struct ListNode* fast = dummy;
    struct ListNode* slow = dummy;

    // 快指针先向前移动n步
    for (int i = 0; i < n; i++) {
        fast = fast->next;
    }

    // fast再提前走一步,因为需要让slow指向删除节点的上一个节点
    fast = fast->next;

    // 快慢指针同时移动,直到快指针到达链表末尾
    while (fast) {
        fast = fast->next;
        slow = slow->next;
    }

    // 删除慢指针的下一个节点(即倒数第n个节点)
    slow->next = slow->next->next;

    // 更新头节点(dummy是虚拟头节点,删除操作不会影响head)
    head = dummy->next;

    // 释放虚拟头节点的内存(注意:这里应该释放dummy,而不是释放dummy->next,因为dummy->next是实际的链表节点)
    free(dummy);

    // 返回更新后的头节点
    return head;
}



3链表相交

给你两个单链表的头节点 headA 和 headB ,请你找出并返回两个单链表相交的起始节点。如果两个链表没有交点,返回 null 。

图示两个链表在节点 c1 开始相交

题目数据 保证 整个链式结构中不存在环。

注意,函数返回结果后,链表必须 保持其原始结构 。

示例 1:

输入:intersectVal = 8, listA = [4,1,8,4,5], listB = [5,0,1,8,4,5], skipA = 2, skipB = 3
输出:Intersected at '8'
解释:相交节点的值为 8 (注意,如果两个链表相交则不能为 0)。
从各自的表头开始算起,链表 A 为 [4,1,8,4,5],链表 B 为 [5,0,1,8,4,5]。
在 A 中,相交节点前有 2 个节点;在 B 中,相交节点前有 3 个节点。

示例 2:

输入:intersectVal = 2, listA = [0,9,1,2,4], listB = [3,2,4], skipA = 3, skipB = 1
输出:Intersected at '2'
解释:相交节点的值为 2 (注意,如果两个链表相交则不能为 0)。
从各自的表头开始算起,链表 A 为 [0,9,1,2,4],链表 B 为 [3,2,4]。
在 A 中,相交节点前有 3 个节点;在 B 中,相交节点前有 1 个节点。

示例 3:

输入:intersectVal = 0, listA = [2,6,4], listB = [1,5], skipA = 3, skipB = 2
输出:null
解释:从各自的表头开始算起,链表 A 为 [2,6,4],链表 B 为 [1,5]。
由于这两个链表不相交,所以 intersectVal 必须为 0,而 skipA 和 skipB 可以是任意值。
这两个链表不相交,因此返回 null 。

提示:

  • listA 中节点数目为 m
  • listB 中节点数目为 n
  • 0 <= m, n <= 3 * 104
  • 1 <= Node.val <= 105
  • 0 <= skipA <= m
  • 0 <= skipB <= n
  • 如果 listA 和 listB 没有交点,intersectVal 为 0
  • 如果 listA 和 listB 有交点,intersectVal == listA[skipA + 1] == listB[skipB + 1]

思路:

  1. 初始化变量:定义 longList 和 shortList 用于存储长链表和短链表的头节点,lenA 和 lenB 用于存储两个链表的长度,gap 用于存储长度差。
  2. 计算链表长度:通过遍历链表 headA 和 headB,分别计算它们的长度 lenA 和 lenB
  3. 确定长链表和短链表:通过比较 lenA 和 lenB,确定哪个是长链表,哪个是短链表,并计算长度差 gap
  4. 尾部对齐:将长链表 longList 移动 gap 步,使得两个链表的尾部对齐。
  5. 同时移动并检查交点:同时移动 longList 和 shortList,检查是否有相同的节点。如果找到相同的节点,则返回该节点(即交点)。
  6. 返回结果:如果没有找到交点,则返回 NULL

代码:

struct ListNode *getIntersectionNode(struct ListNode *headA, struct ListNode *headB) {
    struct ListNode *longList = NULL, *shortList = NULL;
    int lenA = 0, lenB = 0, gap = 0;

    // 求出两个链表的长度
    shortList = headA;
    while (shortList) {
        lenA++;
        shortList = shortList->next;
    }

    shortList = headB;
    while (shortList) {
        lenB++;
        shortList = shortList->next;
    }

    // 确定长链表和短链表,并计算长度差
    if (lenA > lenB) {
        longList = headA;
        shortList = headB;
        gap = lenA - lenB;
    } else {
        longList = headB;
        shortList = headA;
        gap = lenB - lenA;
    }

    // 将长链表和短链表的尾部对齐
    while (gap--) {
        longList = longList->next;
    }

    // 同时移动长链表和短链表,检查是否有相同的节点
    while (longList) {
        if (longList == shortList) return longList; // 找到交点
        longList = longList->next;
        shortList = shortList->next;
    }

    // 没有找到交点
    return NULL;
}

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

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

相关文章

Sqli-labs less-27

Sqli-labs less-27 过滤union\select绕过方式 ### 1. 逻辑绕过 例&#xff1a; 过滤代码 union select user,password from users 绕过方式 1 && (select user from users where userid1)‘admin’### 2.十六进制字符绕过 select ——> selec\x74 union——>un…

AutoFixture:.NET 的假数据生成工具

上次推荐过《Bogus&#xff1a;.NET的假数据生成利器》方便我们制造假数据测试。今天继续推荐另外一个也是非常流行的工具。 01 项目简介 AutoFixture 是一个用于 .NET 的测试工具&#xff0c;它允许开发者在单元测试中自动生成随机的测试数据。它支持广泛的数据类型&#xf…

充电桩高压快充发展趋势

一、为什么要升级充电电压 1、新能源发展的困境 随着电动汽车加快发展&#xff0c;用户对电动汽车接受度不断提高&#xff0c;充电问题是影响电动车普及的重要因素&#xff0c;用户快速补能的需求强烈&#xff0c;例如节假日经常会遇到&#xff0c;高速充电1小时&#xff0c;…

编码方式知识整理【ASCII、Unicode和UTF-8】

编码方式 一、ASCII编码二、Unicode 编码三、UTF-8编码四、GB2312编码五、GBK编码 计算机中对数据的存储为二进制形式&#xff0c;但采用什么样的编码方式存储&#xff0c;效率更高。主要编码方式有 ASCII、Unicode、UTF-8等。 英文一般为1个字节&#xff0c;汉字一般为3个字节…

智联云采 SRM2.0 testService SQL注入漏洞复现

0x01 产品简介 智联云采是一款针对企业供应链管理难题及智能化转型升级需求而设计的解决方案,针对企业供应链管理难题,及智能化转型升级需求,智联云采依托人工智能、物联网、大数据、云等技术,通过软硬件系统化方案,帮助企业实现供应商关系管理和采购线上化、移动化、智能…

Pytorch复习(二)

一、非线性回归 import torch import matplotlib.pyplot as plt from torch import nn,optim from torch.autograd import Variable import numpy as npx_data np.linspace(-2,2,200)[:,np.newaxis] noise np.random.normal(0,0.2,x_data.shape) y_data np.square(x_data) …

.Net自动更新程序GeneralUpdate,适用于wpf,winfrom,控制台应用

GeneralUpdate是基于.net framwork4.5.2开发的一款&#xff08;c/s应用&#xff09;自动升级程序。 第一个版本叫Autoupdate 有人会奇怪为什么会改名称&#xff0c;稍微解释一下是因为在nuget上有重名的项目再者就是新版本更新功能不仅限于wpf程序的更新。 将更新的核心部分抽…

VS Code开发qt项目

没整明白&#xff0c;尴尬 安装扩展 设置cmake路径 前提是已经安装了QT 报错 用msvc选windows启动&#xff0c;用mingw则选gdb启动

pandas处理时间序列-基础入门

公众号&#xff1a;尤而小屋编辑&#xff1a;Peter作者&#xff1a;Peter 大家好&#xff0c;我是Peter~ Pandas 是一个强大的 Python 数据分析库&#xff0c;它提供了非常灵活和高效的方式来处理时间序列数据。 时间序列数据是指按照时间顺序排列的数据点集合&#xff0c;通…

LabVIEW示波器通信及应用

基于LabVIEW平台开发的罗德与施瓦茨示波器通信与应用系统实现了示波器的远程控制及波形数据的实时分析&#xff0c;通过TCP/IP或USB接口与计算机通信&#xff0c;利用VISA技术进行指令传输&#xff0c;从而实现高效的数据采集与处理功能。 项目背景 随着现代电子测试需求的日益…

滑铁卢大学大模型公开课资料来了,大模型入门到精通,非常详细收藏我这一篇就够了

今天给大家推荐的是加拿大滑铁卢大学大模型公开课资源&#xff0c;根据QS世界大学排名&#xff0c;其计算机专业排名全球22。 课程内容主要分为五部分&#xff1a;大模型基础、Transformer架构、大语言模型、多模态大模型、智能体。 每一课除了PPT和视频链接&#xff0c;还有…

013_django基于大数据的高血压人群分析系统2024_dcb7986h_055

目录 系统展示 开发背景 代码实现 项目案例 获取源码 博主介绍&#xff1a;CodeMentor毕业设计领航者、全网关注者30W群落&#xff0c;InfoQ特邀专栏作家、技术博客领航者、InfoQ新星培育计划导师、Web开发领域杰出贡献者&#xff0c;博客领航之星、开发者头条/腾讯云/AW…

VScode实现服务器免密登录(亲测有效)

目录 1 免密步骤1.1 在本地生成密钥1.2 在vscode中下载Remote-SSH1.3 配置SSH文件1.4 在服务器中添加本地公开密钥1.5 远程免密连接试验 2 后记 1 免密步骤 1.1 在本地生成密钥 window R打开命令面板 ssh-keygen1.2 在vscode中下载Remote-SSH 1.3 配置SSH文件 本地密钥的文…

Java 入门基础篇14 - java面向对象思想以及特性

学习目标&#xff1a; 一、目标 面向对象思想类和对象对象的创建和使用属性和方法封装 开始学习&#xff1a; 二、编程思想 2.1 什么是编程思想 做人有做人的原则&#xff0c;编程也有编程的原则。这些编程的原则&#xff0c;就叫做编程思想。 2.2 面向过程和面向对象 二…

卡尔曼讲解与各种典型进阶MATLAB编程(专栏目录,持续更新……)

专栏链接&#xff1a;https://blog.csdn.net/callmeup/category_12574912.html 文章目录 专栏介绍重点文章卡尔曼滤波的原理卡尔曼滤波的例程 进阶MATLAB编程后续更新 专栏介绍 本专栏旨在深入探讨卡尔曼滤波及其在各类应用中的实现&#xff0c;尤其是通过MATLAB编程进行的典…

C++实现 对象移动 移动语义(什么是右值引用)

右值引用 为了支持移动操作&#xff0c;C11增加了右值引用。 使用&&来定义 //int &num100;//错误的使用方法&#xff0c;引用不能引用常量int &&num100;//正确的//同样 int b10; //int &mb*93;//错误的 int &&mb*93;//正确的//当然&#xff0…

什么是DevOps,如何才能获取DevOps相关实践

DevOps 是 “Development”&#xff08;开发&#xff09;和 “Operations”&#xff08;运维&#xff09;的缩写&#xff0c;代表一种融合开发和运维的文化和实践。它强调开发团队与运维团队的协作&#xff0c;通过自动化流程和持续交付&#xff0c;提高软件交付的速度、质量和…

通过Express + Vue3从零构建一个用户认证与授权系统(三)前端应用工程构建

前言 接下来&#xff0c;我们将使用 Vue 3、TypeScript 和 Vite 构建一个前端应用&#xff0c;与之前构建的后端 API 无缝对接。此前端将处理用户认证、显示用户数据、管理角色和权限&#xff0c;并确保与后端的安全通信。首先&#xff0c;我们来构建一个满足基本开发的前端应…

Spring6梳理14——依赖注入之P命名空间

以上笔记来源&#xff1a; 尚硅谷Spring零基础入门到进阶&#xff0c;一套搞定spring6全套视频教程&#xff08;源码级讲解&#xff09;https://www.bilibili.com/video/BV1kR4y1b7Qc 目录 ①搭建模块 ②引入配置文件 ③创建bean-dip.xml文件 ④创建课程类文件 ⑤创建学生…

【C++】string类(2)

&#x1f973;个人主页: 起名字真南 &#x1f973;个人专栏:【数据结构初阶】 【C语言】 【C】 目录 引言1 模拟实现string类基本框架2 实现string类中的主要成员函数2.1 Push_Back 函数2.2 reserve 函数2.3 append 函数2.4 c_str 函数2.5 begin ,end 函数2.5 operator 函数2.6…