【数据结构与算法】单链表的综合运用:1.合并两个有序链表 2.分割链表 3.环形链表的约瑟夫问题

🔥小龙报:个人主页

🎬作者简介:C++研发,嵌入式,机器人等方向学习者

❄️个人专栏:《C语言》《【初阶】数据结构与算法》
永远相信美好的事情即将发生

文章目录

  • 前言
  • 一、合并两个有序链表
    • 1.1题目
    • 1.2 算法原理
    • 1.3代码
  • 二、分割链表
    • 2.1题目
    • 2.2 算法原理
    • 2.3代码
  • 三、环形链表的约瑟夫问题
    • 3.1题目
    • 3.2 算法原理
    • 3.3代码
  • 总结与每日励志

前言

链表是C语言数据结构的核心内容,也是算法面试的高频考点,其灵活的指针操作与逻辑构建对编程思维要求颇高。本文聚焦链表经典实操题型,从合并有序链表、分割链表到环形链表约瑟夫问题,由浅入深拆解解题思路,结合哨兵位、循环计数等实用技巧,通过清晰的算法原理与完整代码实现,帮读者吃透链表操作逻辑,夯实数据结构基础。

一、合并两个有序链表

1.1题目

链接:合并两个有序链表

1.2 算法原理

核心: 判断大小 + 链表为插 + 建立一个哨兵位

和合并两个有序数组的思路一样,定义四个指针

newhead和newtail:新链表的头尾节点

pcur1和pcur2:分别指向两个要合并的链表的头节点
技巧 :可以malloc一块空间让newhead和newtail指向这块空间,使其成为首元节点(哨兵位),这样在插入时可以免去链表为空情况的判断,最后返回新链表的下一个节点即可。
哨兵位: 数值域不存储有效数据,指针域存储有效地址

1.3代码

csharp 复制代码
/**
 * Definition for singly-linked list.
 * struct ListNode {
 *     int val;
 *     struct ListNode *next;
 * };
 */
 typedef struct ListNode ListNode;
struct ListNode* mergeTwoLists(struct ListNode* list1, struct ListNode* list2) 
{
    ListNode* newhead = (ListNode*)malloc(sizeof(ListNode));
    ListNode* pcur1 = list1;;
    ListNode* pcur2 = list2;
    ListNode*newtail = newhead;
    newtail->next = NULL;
    while(pcur1 && pcur2)
    {
        if(pcur1->val <= pcur2->val)
        {
            newtail->next = pcur1;
            newtail = newtail->next;
            pcur1 = pcur1->next;
        }
        else
        {
            newtail->next = pcur2;
            newtail = newtail->next;
            pcur2 = pcur2->next;
        }
    }
    //为遍历完的链表直接接上新链表尾部节点
    if(pcur1)
        newtail->next = pcur1;
     if(pcur2)
        newtail->next = pcur2;
    return newhead->next;
}

注:大家也可以free掉我们手动开辟的节点空间来养成良好的编程习惯,因为这是算法题,在程序执行完后会会自动回收笔者主要讲解思路,就不主动free但大家在写工程时不要的空间要及时释放养成良好编程习惯

二、分割链表

2.1题目

链接:分割链表

2.2 算法原理

创建两个新链表,list1和list2分别存放小于x和大于或等于x的节点,最后让list1的尾指针指向list2的第一个元素即可。
:list2作为新链表的后半部分,最后一个节点的next要及时置NILL,防止死循环
技巧:依旧可以使用哨兵位,并把哨兵位的next初始化为NULL可以避免合并时一条链表为空的特殊判断造成要写大量特判代码。

2.3代码

csharp 复制代码
/**
 * Definition for singly-linked list.
 * struct ListNode {
 *     int val;
 *     struct ListNode *next;
 * };
 */
 typedef struct ListNode ListNode;
struct ListNode* partition(struct ListNode* head, int x) 
{
    if(head == NULL)
        return head;
    //指向小于x链表的头节点
    ListNode* list1 = (ListNode*)malloc(sizeof(ListNode));
    ListNode* tail1 = list1;
     //指向大于x链表的头节点
    ListNode* list2 = (ListNode*)malloc(sizeof(ListNode));
    ListNode* tail2 = list2;
    //防止单个节点链表出现非法解引用
    list1->next = NULL;
    list2->next = NULL;
    ListNode* pcur = head;
    while(pcur)
    {
        if(pcur->val < x)
        {
            tail1->next = pcur;
            tail1 = tail1->next;
        }
        else
        {
            tail2->next = pcur;
            tail2 = tail2->next;
        }
        pcur = pcur->next;
    }
    tail1->next = list2->next;
    //防止死循环
    tail2->next = NULL;

    return list1->next;
}

注:大家也可以free掉我们手动开辟的节点空间来养成良好的编程习惯,因为这是算法题,在程序执行完后会会自动回收笔者主要讲解思路,就不主动free但大家在写工程时不要的空间要及时释放养成良好编程习惯

三、环形链表的约瑟夫问题

3.1题目

链接:环形链表的约瑟夫问题

3.2 算法原理

核心:利用循环链表 + 循环计数

定义一个指针v指尾节点,一个指针指向头结点,利用一个变量s来循环技术,当s == m时让h指向的当前元素出队即可

3.3代码

csharp 复制代码
typedef struct ListNode ListNode;
//创建节点
ListNode* BuyNode(int x)
{
    ListNode* newnode = (ListNode*)malloc(sizeof(ListNode));
    newnode->val = x;
    newnode->next = NULL;
    return newnode;
}
//创建环形链表
ListNode* CyclicList(int n)
{
    ListNode* head = BuyNode(1);
    ListNode* tail = head; 
    for(int i = 2;i <= n;i++)
    {
        tail->next = BuyNode(i);
        tail = tail->next;
    }
    tail->next = head;
    return tail;
}
int ysf(int n, int m ) 
{

   ListNode* prev = CyclicList(n);
   ListNode* phead = prev->next;
   int count = 1;//计数
   while(phead != prev)
   {
    if(count == m)
    {
        prev->next = phead->next;
        free(phead);
        phead = prev->next;
        count = 1;
    }
    else 
    {
         prev = phead;
         phead = phead->next; 
         count++;
    }
   }
   return phead->val;
}

总结与每日励志

✨本文系统讲解了链表操作的三大经典题型:合并有序链表通过哨兵位简化插入逻辑,分割链表采用双链表分类处理,环形链表约瑟夫问题巧妙结合循环计数。每种解法均配有清晰的算法图解和完整代码实现,帮助读者深入理解链表操作的核心逻辑与实用技巧。掌握这些题型不仅能提升面试竞争力,更能培养扎实的数据结构思维。坚持每日练习,保持对算法的热情与专注,终将在编程之路上收获成长与突破。永远相信美好的事情即将发生!

相关推荐
艾莉丝努力练剑几秒前
【Linux:文件 + 进程】进程间通信进阶(1)
linux·运维·服务器·网络·c++·人工智能·进程
珠海西格电力1 分钟前
5G+物联网,零碳园区管理系统的“信息高速路”
大数据·人工智能·物联网·算法·5g
江沉晚呤时1 分钟前
RabbitMQ 延迟队列实战指南:C# 版订单超时与定时任务解决方案
开发语言·后端·ruby
豆浆煮粉1 分钟前
Linux驱动开发理解指针与结构体
linux·c语言·驱动开发
源码师傅2 分钟前
2026最新AI短剧创作系统源码 开发语言:PHP+MYSQL 无限SAAS 含图文搭建教程
开发语言·php·ai短剧创作系统源码·短剧创作系统·短剧创作源码
疋瓞3 分钟前
C\C++\python对比_概览(1)
c语言·c++·python
见叶之秋3 分钟前
【数据结构】详解双向链表
数据结构·链表
海盗猫鸥4 分钟前
「Linux工具」gcc/g++
linux·c语言·c++
Frostnova丶5 分钟前
LeetCode 84 & 85.柱状图最大矩形与最大矩形
算法·leetcode
Yungoal6 分钟前
Visual StudioProfiler对工作流进行热点分析
c++·visual studio