当前位置: 首页 > news >正文

【C语言】递归复杂度与链表OJ之双指针

【C语言】递归复杂度与链表OJ之双指针

🔥个人主页大白的编程日记

🔥专栏数据结构


文章目录

  • 【C语言】递归复杂度与链表OJ之双指针
    • 前言
    • 一.递归复杂度
      • 1.1递归时间复杂度
      • 1.2递归空间复杂度
    • 二.链表OJ之双指针
      • 2.1倒数第K个节点
      • 2.2链表的中间节点
      • 2.3反转链表
      • 2.4回文链表
      • 2.5相交链表
      • 双指针法
    • 后言

前言

哈喽,各位小伙伴大家好!今天我们继续深入探讨递归的复杂度和链表OJ常见的双指针法。话不多说,咱们进入正题!向大厂冲锋!

一.递归复杂度

1.1递归时间复杂度

前面我们讲的复杂度没有涉及到递归,那递归的复杂度又该如何计算呢?

// 计算阶乘递归Fac的时间复杂度?
long long Fac(size_t N)
{if(0 == N)return 1;return Fac(N-1)*N;
}


之前我们都是在一个函数中算时间算复杂度,但是递归会自己调用自己,所以递归的时间复杂度计算就需要把每次函数调用次数累加起来。

那我们现在修改一下代码,现在的时间复杂度又是多少呢?

// 计算阶乘递归Fac的时间复杂度?
long long Fac(size_t N)
{if (0 == N)return 1;for (int i = 0; i < N; i++){;}return Fac(N - 1) * N;
}


函数的调用次数不变,每次函数调用的复杂度改变。将每次函数调用的消耗相加
,时间复杂度函数为等差数列。大O渐进表示法算出O(N^2)。

结论:递归时间复杂度==所有递归递归调用次数累加。

现在我们再来计算一下斐波那契数列的时间复杂度。

// 计算斐波那契递归Fib的时间复杂度?
long long Fib(size_t N)
{if(N < 3)return 1;return Fib(N-1) + Fib(N-2);
}

这里我们需要画出递归展开图,然后发现调用次数是等比数列。
再用错位相减法计算调用次数即可。实际调用次数没那么多。
但是也是O(2^N)的量级。

1.2递归空间复杂度

那递归的复杂度又该如何计算呢?

  • 阶乘
    递归会调用函数,每次调用函数都会开辟栈帧。
    所以递归的空间复杂度是函数调用的次数。
// 计算阶乘递归Fac的空间复杂度?
long long Fac(size_t N)
{if(N == 0)return 1;return Fac(N-1)*N;
}

  • 斐波那契数列
// 计算斐波那契递归Fib的空间复杂度?
long long Fib(size_t N)
{if(N < 3)return 1;return Fib(N-1) + Fib(N-2);
}

斐波那契数列的递归是双倍递归,那他的空间复杂度该如何计算呢?

因为函数栈帧会重复利用,所以斐波那契数列的空间复杂度就是O(N)。

二.链表OJ之双指针

2.1倒数第K个节点

  • 题目
    倒数第k个节点

  • 思路分析


我们先让快慢指针拉开k步,当快指针到NULL节点,慢指针就是倒数第K个节点。

  • 代码
typedef struct ListNode ListNode;
int kthToLast(struct ListNode* head, int k)
{ListNode* slow,*fast;slow=fast=head;while(k--){fast=fast->next;}while(fast){fast=fast->next;slow=slow->next;}return slow->val;
}

2.2链表的中间节点

  • 题目
    链表的中间节点
  • 思路分析
    我们可以定义两个指针,一个每次走一步,一个每次走两步。

这就是我们的快慢指针。两个指针走的路程是2倍关系,当快指针走完时,慢指针也走了快指针的一半路程。

  • 代码
 typedef struct ListNode ListNode;
struct ListNode* middleNode(struct ListNode* head){//创建快慢指针struct ListNode* slow;struct ListNode* fast;slow=fast=head;while(fast&&fast->next){fast=fast->next->next;slow=slow->next;}//此时slow指向中间节点return slow;}

2.3反转链表

  • 题目
    反转链表
  • 思路
  • 我们创建三个节点指针n1,n2,n3。n1刚开始指向NULL,n2指向头节点,n3保存n2的下一个几点。每次都让n2指向n1,然后n1移动到n2,n2移动到n3.n3向后移动。当n2走到尾节点就完成反转。
  • 代码
 typedef struct ListNode ListNode;
struct ListNode* reverseList(struct ListNode* head) {if(head==NULL)return NULL;ListNode* n1=NULL;ListNode* n2=head;ListNode* n3=head->next;while(n2){n2->next=n1;n1=n2;n2=n3;if(n3)n3=n3->next;}return n1;
}

2.4回文链表

  • 题目
    回文链表

  • 思路

  • 我们先找到中间节点,反转以中间节点为头节点的子链表。然后分别从头节点和中间节点开始遍历,一一对比,当中间节点走到NULL时,说明是回文链表。
    在这里插入图片描述

  • 代码

typedef struct ListNode ListNode;struct ListNode* reverseList(struct ListNode* head)
{//创建三个指针ListNode *n1,*n2,*n3;n1=NULL;n2=head;if(n2)n3=n2->next;while(n2){n2->next=n1;n1=n2;n2=n3;if(n3)n3=n3->next;}return n1;
}
struct ListNode* middleNode(struct ListNode* head){//创建快慢指针struct ListNode* slow;struct ListNode* fast;slow=fast=head;while(fast&&fast->next){fast=fast->next->next;slow=slow->next;}//此时slow指向中间节点return slow;}
bool isPalindrome(struct ListNode* head) 
{ListNode* mid=middleNode(head);//找到中间节点ListNode* reverse=reverseList(mid);//反转后面节点ListNode*pcur=head;while(reverse){if(pcur->val!=reverse->val)//匹配{return false;break;}reverse=reverse->next;//移动pcur=pcur->next;}return true;
}

2.5相交链表

  • 题目
    相交链表

  • 思路

  • 判断相交
    分别遍历两个链表到尾节点,判断尾节点地址是否相等。相等就相交

  • 找交点
    让长链表走两链表长度的差距步,此时两节点距离交点差距相等。两节点同时往后遍历,相遇的位置就是交点。

  • 代码
typedef struct ListNode ListNode;
struct ListNode* getIntersectionNode(struct ListNode* headA,struct ListNode* headB) {int lena = 0, lenb = 0;ListNode* pa = headA;ListNode* pb = headB;while (pa->next){lena++;pa = pa->next;//遍历统计}while (pb->next){lenb++;pb = pb->next;//遍历统计}if (pb != pa) {return NULL;}ListNode* pshort = headA;ListNode* plong = headB;if (lena > lenb) //假设法{plong = headA;pshort = headB;}int tmp = abs(lena - lenb);while (tmp--) {plong = plong->next;//走差距步}while (plong!=pshort){pshort = pshort->next;plong = plong->next;}return plong;
}

双指针法

大家发现这些题目都会用到两个指针解决问题,这就是我们说的双指针法,具体如何运用双指针,得看具体题目灵活运用。

后言

这就是递归的复杂度和链表常见OJ。今天就分享到这里,感谢大家耐心垂阅。咱们下期见!拜拜~
在这里插入图片描述

相关文章:

  • 流量暴增如何应对?漏桶限流算法,让你轻松应对流量高峰!揭晓标准代码,超乎想象的稳定、简单!
  • qt仿制qq登录界面
  • 牛客链表刷题(一)
  • I/O Stream设计实验
  • QT 使用资源文件的注意点
  • C# 通过Path获取后缀,文件名,目录等
  • JAVA学习笔记DAY5——Spring_Ioc
  • 李诞-2021.8脱口秀工作手册-1-工作的本质是交易;脱口秀是一份和生活分不开的工作,你的全部人生都理应要为你的创作提供养分,为它服务。
  • 使用CSS、JavaScript、jQuery三种方式实现手风琴效果
  • AIGC全面介绍
  • 【数据结构陈越版笔记】进阶实验1-3.1:两个有序序列的中位数
  • Android APP memory统计方法
  • 2024-06-07 Unity 编辑器开发之编辑器拓展8 —— Scene 窗口拓展
  • 【C++】模板及模板的特化
  • MySQL的索引类型,以及各自的作用
  • 【React系列】如何构建React应用程序
  • docker-consul
  • java取消线程实例
  • SpringCloud集成分布式事务LCN (一)
  • VuePress 静态网站生成
  • 读懂package.json -- 依赖管理
  • 聚簇索引和非聚簇索引
  • 浅谈web中前端模板引擎的使用
  • 推荐一款sublime text 3 支持JSX和es201x 代码格式化的插件
  • 运行时添加log4j2的appender
  • Hibernate主键生成策略及选择
  • 曜石科技宣布获得千万级天使轮投资,全方面布局电竞产业链 ...
  • ​​​​​​​​​​​​​​汽车网络信息安全分析方法论
  • #《AI中文版》V3 第 1 章 概述
  • #大学#套接字
  • (1)虚拟机的安装与使用,linux系统安装
  • (2024,Flag-DiT,文本引导的多模态生成,SR,统一的标记化,RoPE、RMSNorm 和流匹配)Lumina-T2X
  • (C语言)strcpy与strcpy详解,与模拟实现
  • (大众金融)SQL server面试题(1)-总销售量最少的3个型号的车及其总销售量
  • (剑指Offer)面试题34:丑数
  • (三)docker:Dockerfile构建容器运行jar包
  • (原創) 人會胖會瘦,都是自我要求的結果 (日記)
  • (转载)(官方)UE4--图像编程----着色器开发
  • ./indexer: error while loading shared libraries: libmysqlclient.so.18: cannot open shared object fil
  • .NET Compact Framework 3.5 支持 WCF 的子集
  • .NET Framework杂记
  • .NET 中 GetHashCode 的哈希值有多大概率会相同(哈希碰撞)
  • .NET/C# 如何获取当前进程的 CPU 和内存占用?如何获取全局 CPU 和内存占用?
  • .Net8 Blazor 尝鲜
  • .NET编程C#线程之旅:十种开启线程的方式以及各自使用场景和优缺点
  • .NET实现之(自动更新)
  • .NET下的多线程编程—1-线程机制概述
  • .NET项目中存在多个web.config文件时的加载顺序
  • .vimrc 配置项
  • @CacheInvalidate(name = “xxx“, key = “#results.![a+b]“,multi = true)是什么意思
  • [ Python ]使用Charles对Python程序发出的Get与Post请求抓包-解决Python程序报错问题
  • [C++][ProtoBuf][初识ProtoBuf]详细讲解
  • [C++][基础]1_变量、常量和基本类型
  • [GN] DP学习笔记板子
  • [GXYCTF2019]禁止套娃