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

embedded/2024/9/23 4:26:26/

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

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

🔥专栏数据结构


文章目录

  • 【C语言】递归复杂度与链表OJ之双指针
    • 前言
    • 一.递归复杂度
      • 1.1递归时间复杂度
      • 1.2递归空间复杂度
    • 二.链表OJ之双指针
    • 后言

前言

哈喽,各位小伙伴大家好!今天我们继续深入探讨递归的复杂度和链表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。今天就分享到这里,感谢大家耐心垂阅。咱们下期见!拜拜~
在这里插入图片描述


http://www.ppmy.cn/embedded/48669.html

相关文章

ABBYY Finereader 15软件下载及安装教程

ABBYY FineReader 是一款功能强大的 OCR&#xff08;Optical Character Recognition&#xff09;软件&#xff0c;可以将扫描的文档转换为可编辑的文本文件。它不仅可以识别文本&#xff0c;还可以识别表格、图像和布局&#xff0c;使得文档的转换更加准确和方便。 安 装 包 获…

webrtc新版本无法连接peerconnection_server、无法音视频互通no incoming video...问题解决

问题1:无法连接peerconnection_server 在webrtc大概2022之后的版本,会出现无法连接peerconnection_server的现象,如下图: 在peerconnection_client界面点击Connect无法连接server. 解决办法 我们需要修改peerconnection_client的main.cc代码,如下图: 新添加的类代码…

算法:位运算

前言 数据结构和算法是一个程序员的必过的两道门槛&#xff0c;前面我们把常见的数据结构进行了详细的介绍和实现。本专栏将进行学习常见的算法&#xff01; 本期内容介绍 位运算常见的操作总结 位运算在OJ中的使用解析 位运算常见的操作总结 位运算基础 << 左移&…

尝试用 GPT-4o 写 2024高考语文作文

文章目录 新课标I卷科技进步与问题的演变 新课标II卷抵达未知之境&#xff1a;探索与成长的旅程 全国甲卷坦诚交流&#xff1a;构建真正相遇的桥梁 北京卷历久弥新 天津卷定义与自定义&#xff1a;在世界的缤纷中前行 上海卷认可度的思考与反思 新课标I卷 阅读下面的材料&#…

RV32F\RV32D指令集

RV32F\RV32D指令集 F扩展1、浮点控制状态寄存器2、指令类型F扩展 F扩展增加了32个浮点寄存器f0-f31,每个32位宽,以及一个浮点控制和状态寄存器fcsr,其中包含浮点单元的工作模式和异常状态。FLEN=32表示F单精度浮点扩展,大多数浮点指令对浮点寄存器中的值进行操作。浮点加载…

Mysql5.7安装教程(详细图解教程)_mysql5.7下载

本文讲解的是mysql5.7安装包、mysql5.7下载、mysql5.7安装配置教程、离线安装mysql5.7。MySQL 5.7 是 MySQL 数据库的一个重要版本&#xff0c;它引入了许多新特性和改进&#xff0c;旨在提高性能、安全性和易用性。 MySQL 5.7 在所有负载模型上都有显著的性能改进&#xff0c…

ckplayer去掉倍速和清晰度按钮

ckplayer X2 去掉倍速和清晰度按钮 1 去掉倍速&#xff1a;在ckplayer.js文件中&#xff0c;ckplayerConfig方法中config属性&#xff0c;playbackRate设置为false即可 2 去掉清晰度&#xff1a;配置视频url是配置单一清晰度 例如&#xff1a; //多清晰度that.videoObject.vi…

Android输入法IME(四)之 服务端(IMS)启动流程

2.3. IME服务端&#xff08;IMS&#xff09;初始化流程 IMS运行在输入法进程, 是一种特殊的输入法后台服务&#xff0c;继承结构为:InputMethodService extends AbstractInputMethodServiceService 输入法服务本质上是一个服务, 使用时需要IMMS通过bindService的方式绑定。 初…