链表 —— 常用技巧与操作总结详解

引言

        链表作为一种动态数据结构,以其灵活的内存管理和高效的插入删除操作,在算法与工程实践中占据重要地位。然而,链表的指针操作复杂,容易引发内存泄漏和野指针问题。本文博主将从基础操作到高阶技巧,系统化解析链表的常用操作与优化方法,结合C++代码示例与复杂度分析,深入理解链表的实现细节与应用场景。无论是解决经典算法问题,还是优化实际工程性能,掌握这些技巧都将事半功倍。

目录

引言

一、链表基本结构与类型

1. 单链表节点定义

2. 双链表节点定义

3. 循环链表特性

二、基础操作与实现

1. 头插法构建链表

2. 尾插法构建链表

3. 指定位置插入

4. 节点删除操作

三、高阶操作技巧

1. 快慢指针应用

场景1:检测环形链表

场景2:寻找中间节点

2. 链表反转

迭代法:

递归法:

3. 合并有序链表

4. 链表排序(归并排序实现)

四、内存管理要点

1. 智能指针应用

2. 手动释放链表

3. 野指针处理

五、特殊场景处理

1. 哑节点(Dummy Node)技巧

2. 多指针协同操作

六、复杂度对比与优化

七、调试与测试技巧

1. 可视化打印链表

2. 边界测试用例

八、工程实践建议


一、链表基本结构与类型

1. 单链表节点定义

struct ListNode {
    int val;
    ListNode* next;
    ListNode(int x) : val(x), next(nullptr) {}
};

2. 双链表节点定义

struct DListNode {
    int val;
    DListNode *prev, *next;
    DListNode(int x) : val(x), prev(nullptr), next(nullptr) {}
};

3. 循环链表特性

  • 尾节点指向头节点形成闭环

  • 适用于环形缓冲区、约瑟夫问题等场景


二、基础操作与实现

1. 头插法构建链表

ListNode* createListHeadInsert(vector<int>& nums) {
    ListNode* dummy = new ListNode(-1);
    for (int num : nums) {
        ListNode* node = new ListNode(num);
        node->next = dummy->next;
        dummy->next = node;
    }
    return dummy->next; // 实际头节点
}
// 示例:输入[1,2,3],生成3->2->1

2. 尾插法构建链表

ListNode* createListTailInsert(vector<int>& nums) {
    ListNode* dummy = new ListNode(-1);
    ListNode* tail = dummy;
    for (int num : nums) {
        tail->next = new ListNode(num);
        tail = tail->next;
    }
    return dummy->next;
}
// 保持原始顺序的关键技巧

3. 指定位置插入

void insertNode(ListNode* prevNode, int val) {
    if (!prevNode) return;
    ListNode* newNode = new ListNode(val);
    newNode->next = prevNode->next;
    prevNode->next = newNode;
}
// 时间复杂度:O(1),但需提前定位prevNode

4. 节点删除操作

void deleteNode(ListNode* &head, int target) {
    ListNode dummy(0);
    dummy.next = head;
    ListNode* curr = &dummy;
    
    while (curr->next) {
        if (curr->next->val == target) {
            ListNode* temp = curr->next;
            curr->next = temp->next;
            delete temp;
        } else {
            curr = curr->next;
        }
    }
    head = dummy.next;
}
// 使用哑节点统一处理头节点删除

三、高阶操作技巧

1. 快慢指针应用

场景1:检测环形链表

bool hasCycle(ListNode* head) {
    ListNode *slow = head, *fast = head;
    while (fast && fast->next) {
        slow = slow->next;
        fast = fast->next->next;
        if (slow == fast) return true;
    }
    return false;
}
// Floyd判圈算法,时间复杂度O(n)

场景2:寻找中间节点

ListNode* findMiddle(ListNode* head) {
    ListNode *slow = head, *fast = head;
    while (fast && fast->next && fast->next->next) {
        slow = slow->next;
        fast = fast->next->next;
    }
    return slow;
}
// 快指针到达末尾时,慢指针指向中间

2. 链表反转

迭代法:

ListNode* reverseList(ListNode* head) {
    ListNode *prev = nullptr, *curr = head;
    while (curr) {
        ListNode* nextTemp = curr->next;
        curr->next = prev;
        prev = curr;
        curr = nextTemp;
    }
    return prev;
}
// 时间复杂度O(n),空间O(1)

递归法:

ListNode* reverseListRecursive(ListNode* head) {
    if (!head || !head->next) return head;
    ListNode* p = reverseListRecursive(head->next);
    head->next->next = head;
    head->next = nullptr;
    return p;
}
// 栈深度O(n),需注意栈溢出风险

3. 合并有序链表

ListNode* mergeTwoLists(ListNode* l1, ListNode* l2) {
    ListNode dummy(0);
    ListNode* tail = &dummy;
    
    while (l1 && l2) {
        if (l1->val < l2->val) {
            tail->next = l1;
            l1 = l1->next;
        } else {
            tail->next = l2;
            l2 = l2->next;
        }
        tail = tail->next;
    }
    tail->next = l1 ? l1 : l2;
    return dummy.next;
}
// 时间复杂度O(m+n),空间O(1)

4. 链表排序(归并排序实现)

ListNode* sortList(ListNode* head) {
    if (!head || !head->next) return head;
    
    ListNode* mid = findMiddle(head);
    ListNode* right = mid->next;
    mid->next = nullptr;
    
    ListNode* left = sortList(head);
    right = sortList(right);
    return mergeTwoLists(left, right);
}
// 综合应用找中点、分割、合并技巧

四、内存管理要点

1. 智能指针应用

struct SafeListNode {
    int val;
    shared_ptr<SafeListNode> next;
    SafeListNode(int x) : val(x), next(nullptr) {}
};
// 自动内存回收,避免泄漏

2. 手动释放链表

void deleteList(ListNode* head) {
    while (head) {
        ListNode* temp = head;
        head = head->next;
        delete temp;
    }
}
// 必须显式调用防止内存泄漏

3. 野指针处理

  • 删除节点后立即置空指针

  • 使用nullptr初始化未使用的指针


五、特殊场景处理

1. 哑节点(Dummy Node)技巧

ListNode* removeElements(ListNode* head, int val) {
    ListNode dummy(0);
    dummy.next = head;
    ListNode* curr = &dummy;
    
    while (curr->next) {
        if (curr->next->val == val) {
            ListNode* temp = curr->next;
            curr->next = temp->next;
            delete temp;
        } else {
            curr = curr->next;
        }
    }
    return dummy.next;
}
// 统一处理头节点与其他节点

2. 多指针协同操作

void reorderList(ListNode* head) {
    if (!head || !head->next) return;
    
    // 找中点并分割
    ListNode* mid = findMiddle(head);
    ListNode* l2 = mid->next;
    mid->next = nullptr;
    
    // 反转后半部分
    l2 = reverseList(l2);
    
    // 交替合并
    ListNode* l1 = head;
    while (l1 && l2) {
        ListNode* next1 = l1->next;
        ListNode* next2 = l2->next;
        
        l1->next = l2;
        l2->next = next1;
        
        l1 = next1;
        l2 = next2;
    }
}
// L0→L1→...→Ln → L0→Ln→L1→Ln-1...

六、复杂度对比与优化

操作常规实现复杂度优化方法
查找元素O(n)跳表结构优化至O(log n)
插入/删除(已知位置)O(1)使用哈希表维护节点位置
反转整个链表O(n)迭代法优于递归法空间复杂度
检测环O(n)Brent算法优化常数因子

七、调试与测试技巧

1. 可视化打印链表

void printList(ListNode* head) {
    while (head) {
        cout << head->val;
        if (head->next) cout << "->";
        head = head->next;
    }
    cout << "->NULL" << endl;
}
// 输出格式:1->2->3->NULL

2. 边界测试用例

  • 空链表

  • 单节点链表

  • 完全逆序链表

  • 含环链表

  • 超长链表(测试栈溢出)


八、工程实践建议

  1. 模块化设计:分离链表操作与业务逻辑

  2. 防御性编程

    • 检查空指针访问

    • 验证节点关联关系

  3. 性能监控

    • 统计链表操作耗时

    • 检测内存泄漏(Valgrind工具)

  4. 容器选择原则

    • 频繁随机访问 → 选择数组

    • 频繁插入删除 → 选择链表


通过系统掌握这些技巧,可高效解决如下经典问题:

  • LRU缓存实现(双链表+哈希表)

  • 多项式运算(链表表示稀疏多项式)

  • 大整数运算(每位数字存储于链表节点)

  • 图邻接表表示

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Renesas RH850 FDL库介绍

文章目录 FDL库(Data Flash Library)简介FDL库的核心功能FDL库的使用步骤关键注意事项示例应用场景总结FDL库(Data Flash Library)简介 FDL(Data Flash Library)是Renesas为RH850系列微控制器提供的数据闪存(Data Flash)操作库,用于简化数据闪存的擦除、写入、读取等…

Linux 配置 MySQL 定时自动备份到另一台服务器

Linux 配置 MySQL 定时自动备份到另一台服务器 前言1、配置服务器通信1.1&#xff1a;配置过程 2、编写自动备份sh脚本文件3&#xff1a;设置定时自动执行 前言 此方案可使一台服务器上的 MySQL 中的所有数据库每天 0 点自动转储为 .sql 文件&#xff0c;然后将文件同步到另一…

用php tp6对接钉钉审批流的 table 表格 明细控件 旧版sdk

核心代码 foreach ($flows[product_list] as $k>$gift) {$items_list[] [[name > 商品名称, value > $gift[product_name] ?? ],[name > 规格, value > $gift[product_name] ?? ],[name > 数量, value > $gift[quantity] ?? ],[name > 单位, v…

RV1126解码(1)

比如我们现在要拉一个流&#xff0c; 拉一个rtmp或者拉一个rtsp的流&#xff0c;让它显示到显示屏上面去&#xff0c;此时就要用到我们这个解码模块了&#xff0c;把它个解出来并且发到其他模块去。 主要功能是通过FFMPEG的API读取每一帧的音视频数据&#xff0c;并通过RV1126的…

sql:时间盲注和boolen盲注

关于时间盲注&#xff0c;boolen盲注的后面几个获取表、列、具体数据的函数补全 时间盲注方法 import time import requests# 获取数据库名 def inject_database(url):dataname for i in range(1, 20):low 32high 128mid (low high) // 2while low < high:payload &q…

DeepSeek+Excel 效率翻倍

2025年初&#xff0c;DeepSeek以惊人的效率突破技术壁垒&#xff0c;用极低的成本实现了与行业顶尖AI相媲美的性能&#xff0c;瞬间成为全球科技领域的热门话题。 那么AI工具的普及将如何改变我们的工作方式&#xff1f;Excel会被取代吗&#xff1f; 今天&#xff0c;珠珠带你…

WPS或word接入智能AI

DeepSeek接入WPS 配置WPS &#xff08;1&#xff09;下载 OfficeAl助手插件: 插件下载地址:https://www.office-ai.cn/。 安装插件后&#xff0c;打开WPS&#xff0c;菜单栏会新增"OfficeAl助手”选项卡。 如果没有出现&#xff0c; 左上找到文件菜单 -> 选项 ,在…

论文学习记录之《CLR-VMB》

目录 一、基本介绍 二、介绍 三、方法 3.1 FWI中的数据驱动方法 3.2 CLR-VMB理论 3.3 注意力块 四、网络结构 4.1 网络架构 4.2 损失函数 五、实验 5.1 数据准备 5.2 实验设置 5.3 训练和测试 5.4 定量分析 5.5 CLR方案的有效性 5.6 鲁棒性 5.7 泛化性 六、讨…

使用 EDOT 监测由 OpenAI 提供支持的 Python、Node.js 和 Java 应用程序

作者&#xff1a;来自 Elastic Adrian Cole Elastic 很自豪地在我们的 Python、Node.js 和 Java EDOT SDK 中引入了 OpenAI 支持。它们为使用 OpenAI 兼容服务的应用程序添加日志、指标和跟踪&#xff0c;而无需任何代码更改。 介绍 去年&#xff0c;我们宣布了 OpenTelemetry…

Golang的多团队协作编程模式与实践经验

Golang的多团队协作编程模式与实践经验 一、多团队协作编程模式概述 在软件开发领域&#xff0c;多团队协作编程是一种常见的工作模式。特别是对于大型项目来说&#xff0c;不同团队间需要协同合作&#xff0c;共同完成复杂的任务。Golang作为一种高效、并发性强的编程语言&…

Sequence to Sequence model

基础模型 基础模型是用RNN模型&#xff0c;前部分是encoder用来寻找法语输入的编码&#xff0c;后半部分是decoder用来生成英文翻译作为输出&#xff0c;每次输出一个单词&#xff0c;直到输出结束标志如EOS。 下面是另一个例子&#xff0c;在CNN模型输出层之前会输出图片的向…

verilog练习:i2c slave 模块设计

文章目录 前言1.结构2.代码2.1 iic_slave.v2.2 sync.v2.3 wr_fsm.v2.3.1 状态机状态解释 2.4 ram.v 3. 波形展示4. 建议5. 资料总结 前言 首先就不啰嗦iic协议了&#xff0c;网上有不少资料都是叙述此协议的。 下面将是我本次设计的一些局部设计汇总&#xff0c;如果对读者有…

【竞技宝】PGL瓦拉几亚S4预选:Tidebound2-0轻取spiky

北京时间2月13日,DOTA2的PGL瓦拉几亚S4预选赛继续进行,昨日进行的中国区预选赛胜者组首轮Tidebound对阵的spiky比赛中,以下是本场比赛的详细战报。 第一局: 首局比赛,spiky在天辉方,Tidebound在夜魇方。阵容方面,spiky点出了幻刺、火枪、猛犸、小强、巫妖,Tidebound则是拿到飞…

Android RenderEffect对Bitmap高斯模糊(毛玻璃),Kotlin(1)

Android RenderEffect对Bitmap高斯模糊(毛玻璃)&#xff0c;Kotlin&#xff08;1&#xff09; import android.graphics.Bitmap import android.graphics.BitmapFactory import android.graphics.HardwareRenderer import android.graphics.PixelFormat import android.graphic…

AI前端开发的崛起与ScriptEcho的助力

近年来&#xff0c;人工智能&#xff08;AI&#xff09;技术飞速发展&#xff0c;深刻地改变着软件开发的格局。尤其是在前端开发领域&#xff0c;AI的应用越来越广泛&#xff0c;催生了对AI写代码工具的需求激增&#xff0c;也显著提升了相关人才的市场价值。然而&#xff0c;…

【Mac排错】ls: command not found 终端命令失效的解决办法

【TroubleShooting on Mac】ls: command not found 终端命令失效的解决办法 A Solution to Solve “Command not found” of Terminal on Mac 一直在使用心爱的MacBook Pro的Terminal&#xff0c;并且为她定制了不同的Profile。 这样&#xff0c;看起来她可以在不同季节&…

DexVLA:通用机器人控制中具有插件式扩散专家的视觉语言模型

25年2月来自美的集团和华东师范的论文“DexVLA: Vision-Language Model with Plug-In Diffusion Expert for General Robot Control”。 让机器人能够在不同的环境中执行不同的任务是机器人学习的核心挑战。虽然视觉-语言-动作 (VLA) 模型已显示出可泛化机器人技能的前景&…

【微服务学习一】springboot微服务项目构建以及nacos服务注册

参考链接 3. SpringCloud - 快速通关 springboot微服务项目构建 教程中使用的springboot版本是3.x&#xff0c;因此需要使用jdk17&#xff0c;并且idea也需要高版本&#xff0c;我这里使用的是IDEA2024。 环境准备好后我们就可以创建springboot项目&#xff0c;最外层的项目…

DeepSeek 助力 Vue 开发:打造丝滑的返回顶部按钮(Back to Top)

前言&#xff1a;哈喽&#xff0c;大家好&#xff0c;今天给大家分享一篇文章&#xff01;并提供具体代码帮助大家深入理解&#xff0c;彻底掌握&#xff01;创作不易&#xff0c;如果能帮助到大家或者给大家一些灵感和启发&#xff0c;欢迎收藏关注哦 &#x1f495; 目录 Deep…

deepseek大模型,本地搭建deepseek模型,springai调用本地deepseek模型,java调用deepseek大模型api

文档对应的视频地址&#xff1a; https://www.bilibili.com/video/BV1V8NBevEjk/?spm_id_from333.1387.homepage.video_card.click&vd_source14d27ec13a4737c281b7c79463687112SpringAI调用本地deepseek模型 一、 使用deepseek步骤 官网注册账号 地址&#xff1a; https…