单链表经典算法

一,移除链表元素

思路一

遍历数组,如果遇到链表中的元素等于val的节点就执行删除操作

cpp 复制代码
typedef struct ListNode ListNode;
 
 struct ListNode* removeElements(struct ListNode* head, int val) {
  
    if(head==NULL)
    {
        return NULL;
    } 
     ListNode*pnewhead=(ListNode*)malloc(sizeof(ListNode));
    pnewhead->next=head;
    ListNode*pcur=head;
    ListNode*prev=pnewhead;
    while(pcur)
    {
        if(pcur->val==val)
        {
            prev->next=pcur->next;
            free(pcur);
            pcur=NULL;
            pcur=prev->next;
        }
        else
        {
            prev=pcur;
            pcur=pcur->next;
            
        }
    }
    return pnewhead->next;
 }

思路二

建立一个新的链表,遇到的节点里面存的数据如果不等于val的值的话就将这个节点尾插到建立的新的链表的后面

cpp 复制代码
typedef struct ListNode ListNode;
 struct ListNode*removeElement(struct ListNode*head,int val)
 {
    ListNode*pcur=head;
    ListNode*newhead,*newtail;//建立新的链表
    newhead=newtail=NULL;
    while(pcur)//遍历链表
    {
        if(pcur->val!=val)
        {
            if(newhead==NULL)
            {
                newhead=newtail=pcur;//新链表为空,将当前的节点赋值给新的链表
            }else
            {
                newtail->next=pcur;//不为空,插到新链表的后面
                newtail=newtail->next;
            }
        }
        pcur=pcur->next;
    }
    if(newtail)//如果链表的不为空的话,就让尾节点的后面为空
    {
        newtail->next=NULL;
    }
    return newhead;
 }
 

代码说明

cpp 复制代码
 if(newtail)//如果链表的不为空的话,就让尾节点的后面为空
 {
     newtail->next=NULL;
 }

因为链表在插入的时候,当前节点的后面还有节点,插入的时候会后面的节点会跟在后面,所以要将原链表的尾节点的后面的节点置为空,比如下面的

当要将上面的链表中的4处于的节点的位置插入到下面0位于的节点的后面,插入的不是4,而是4和4链接的5位于的节点,插入的是后面的一串,这和顺序表不一样

二.反转链表

思路

这里可以定义三个指针,一个指向当前节点,一个指向当前节点的上一个节点,一个指向当前节点的下一个节点,直接让当前节点指向上一个节点

cpp 复制代码
typedef struct ListNode ListNode;
struct ListNode* reverseList(struct ListNode* head) {
    ListNode*prev=NULL;//前驱节点
    ListNode*pcur=head;//当前节点
    ListNode*back=NULL;//当前节点的下一个节点
     if(head==NULL)
    {
        return NULL;
    }
   
    if(head->next==NULL)
    {
        return head;
    }
    while(pcur)
    {
        pcur->next=prev;
         prev=pcur;
        pcur=back;
        if(back!=NULL){
        back=back->next;
        }
    }
    return prev;
}

三.将两个升序链表合并成为一个新的升序链表

思路:定义一个新的无效链表头(里面没有数据,在后续的程序中不需要判断该链表是否是空链表,能够对代码进行优化,有返回值的时候直接返回这个链表头的下一个节点就可以了),以及两个指针,分别指向两个升序链表,比较两个链表里面的值,谁小就把谁放到新链表的后面,然后这个指针指向下一个节点就行了

cpp 复制代码
typedef struct ListNode ListNode;
struct ListNode* mergeTwoLists(struct ListNode* list1, struct ListNode* list2) {
    ListNode*pcur1=list1;//定义指针1
    ListNode*pcur2=list2;//定义指针2
    if(list1==NULL)
    {
        return list2;//为空链表的话返回另一个链表
    }
    if(list2==NULL)
    {
        return list1;//为空的话返回另一个链表
    }
    ListNode*phead=(ListNode*)malloc(sizeof(ListNode));
    ListNode*pcur3=phead;
    while(pcur1&&pcur2)
    {
        if(pcur1->val<=pcur2->val)
        {
            pcur3->next=pcur1;
            pcur3=pcur3->next;
            pcur1=pcur1->next;
        }
        else
        {
             pcur3->next=pcur2;
            pcur3=pcur3->next;
            pcur2=pcur2->next;
        }
    }
    if(pcur2==NULL)
    {
        pcur3->next=pcur1;//插入的是该节点以及该节点后面的节点,一步到位
    }
    if(pcur1==NULL)
    {
       
       pcur3->next=pcur2;
    }
    return phead->next;

}

四.链表的中间节点

思路:定义两个指针,一个快指针,一个慢指针,满指针一次走一格,快指针一次走两格,最后返回慢指针就行了

cpp 复制代码
typedef struct ListNode ListNode;
struct ListNode* middleNode(struct ListNode* head) {
    ListNode*fast,*slow;
    fast=head,slow=head;
    if(head->next==NULL)
    {
        return head;
    }
    while(fast&&fast->next)
    {
        slow=slow->next;
        
        fast=fast->next->next;
            
        
    }
    return slow;
    
}

五,分割链表

思路:定义两个新的链表,遍历原链表,原链表里面的节点里面的值和x比较,小的尾插在第一个链表后面,大于等于的尾插在第二个链表的后面,最后连接两个链表

cpp 复制代码
typedef struct ListNode ListNode;
struct ListNode* partition(struct ListNode* head, int x){
  ListNode*pcur=head;
  ListNode*pcur1=(ListNode*)malloc(sizeof(ListNode));
  ListNode*pcur2=(ListNode*)malloc(sizeof(ListNode));
  ListNode*head1=pcur1;
  ListNode*head2=pcur2;
  if(head==NULL)
  {
    return NULL;
  }
  if(head->next==NULL)
  {
    return head;
  }

  while(pcur)
  {
    if(pcur->val<x)
    {
        pcur1->next=pcur;
        pcur1=pcur1->next;
    }
    else
    {
        pcur2->next=pcur;
        pcur2=pcur2->next;
    }
    pcur=pcur->next;
  }
  pcur2->next=NULL;
  pcur1->next=head2->next;
 return head1->next;
}

注意:上面的代码为什么要有下面的这句话嘞?

pcur2->next=NULL;

原因是插入链表节点的时候,后面带着的节点是跟在本节点的后面的,插的时候就顺带着了,本节点后面不置为空就会造成死循环

六.环形链表的约瑟夫问题

什么是环形链表的约瑟夫问题,先看一看下面的故事:

著名的Josephus问题 据说著名犹太 历史学家 Josephus有过以下的故事:在罗⻢⼈占领乔塔帕特后,39个犹太⼈与 Josephus及他的朋友躲到⼀个洞中,39个犹太⼈决定宁愿死也不要被⼈抓到,于是决定了⼀个⾃杀 ⽅式,41个⼈排成⼀个圆圈,由第1个⼈开始报数,每报数到第3⼈该⼈就必须⾃杀,然后再由下⼀ 个重新报数,直到所有⼈都⾃杀⾝亡为⽌。 然⽽Josephus和他的朋友并不想遵从,Josephus要他的朋友先假装遵从,他将朋友与⾃⼰安排在 第16个与第31个位置,于是逃过了这场死亡游戏。

第一步:创建一个环形链表

第二部:定义一个变量,初值为1,如果该变量不等于m,该变量加1,继续遍历链表,等于,删除该节点,变量重新等于1

cpp 复制代码
typedef struct ListNode ListNode;
//产生一个节点
 ListNode*buynode(int x)
 {
    ListNode*newnode=(ListNode*)malloc(sizeof(ListNode));
    newnode->val=x;
    newnode->next=NULL;
    return newnode;
 }
//建立一个环形链表
 ListNode*creatcircle(int n)
 {
    ListNode*phead=buynode(1);
    ListNode*pcur=phead;
    phead->val=1;
    for(int i=2;i<=n;i++)
    {
        pcur->next=buynode(i);
        pcur=pcur->next;
        
    }
   pcur->next=phead;
   return phead;
 }
int ysf(int n, int m )
 {
   
    ListNode*phead=creatcircle(n);
    ListNode*pcur=phead;
    ListNode*prev=NULL;
    int count=1;
    while(pcur->next!=pcur)
    {
        if(count==m)
        {
            prev->next=pcur->next;
            free(pcur);
            pcur=NULL;
            count=1;
            pcur=prev->next;
        }
        else 
        {
            prev=pcur;
            pcur=pcur->next;
            count++;
        }
        
    }
    return pcur->val;
}

七.结语

在链表中,最值得注意的是,尾插链表,插入的是该节点和该节点后面的节点,也就是一串节点

在插入链表的时候,记得要将该节点后面置为空

相关推荐
打不了嗝 ᥬ᭄1 小时前
P3884 [JLOI2009] 二叉树问题
数据结构·算法·蓝桥杯
earthzhang20213 小时前
《深入浅出HTTPS》读书笔记(28):DSA数字签名
开发语言·网络协议·算法·https·1024程序员节
Lulsj3 小时前
代码随想录day28 | leetcode 56.合并区间 738.单调自增的数字 968.监控二叉树
数据结构·算法·leetcode
get_money_4 小时前
动态规划汇总1
开发语言·数据结构·笔记·算法·leetcode·动态规划·代理模式
Huazzi.4 小时前
【算法学习】——整数划分问题详解(动态规划)
开发语言·数据结构·c++·学习·算法·动态规划
BingLin-Liu4 小时前
备战蓝桥杯:树的存储与遍历(dfs和bfs)
算法·深度优先·宽度优先
AI技术控5 小时前
计算机视觉算法实战——头盔检测
人工智能·算法·计算机视觉
用手码出世界7 小时前
二分查找算法——搜索插入位置
数据结构·算法·leetcode
得物技术7 小时前
Triton-Lang在Transformer优化加速中的实践 | 得物技术
人工智能·算法
低调学习者7 小时前
每日一题(三):压缩字符串(行程长度编码)
数据结构·算法