算法——递归

目录

汉诺塔问题

合并两个有序链表

反转链表

两两交换链表中的节点

[Pow(x, n)](#Pow(x, n))

汉诺塔问题

思路:

结合上图,我们可以定义一个函数,假设函数头为 dfs (x, y, z, n),这个函数的作用是将 x 柱子上的 n 个盘子,借助 y 柱子,转移到 z 柱子上面,那么上图整个流程可以表示如下:(假设图中x柱子上一共 n 个盘子)

  1. dfs (x, z, y, n - 1):将 x 柱子上面的 n - 1 个盘子借助 z 柱子,移动到 y 柱子上。
  2. x.back() --> z:将 x 柱子上最下面的那个盘子直接移动到 z 柱子上。
  3. dfs (y, x, z, n - 1):将 y 柱子上剩余的 n - 1 个盘子借助 x 移动到 z 上。

递归出口:当只剩下一个盘子(即 n = 1)时,就无需再借助其他柱子移动了,直接移动到目标柱子上即可。

代码:

cpp 复制代码
class Solution {
public:
    void hanota(vector<int>& A, vector<int>& B, vector<int>& C) {
        dfs(A, B, C, A.size());
    }
    void dfs(vector<int>& A, vector<int>& B, vector<int>& C, int n){
        if(n == 1){
            C.push_back(A.back());
            A.pop_back();
            return;
        }

        dfs(A, C, B, n - 1);
        C.push_back(A.back());
        A.pop_back();
        dfs(B, A, C, n - 1);
    }
};

合并两个有序链表

**思路:**合并两个有序链表,本质上就是每次只选当前两个链表头中更小的那个节点,然后让这个节点的next指向 "剩下的两个子链表合并后的结果"。

我们写一个递归函数,作用是合并两个链表,然后模拟上述说的过程即可,先找到小的节点,然后让这个节点的next指向 "剩下的两个子链表合并后的结果",因为递归函数的作用就是合并链表,所以剩下两个子链表合并的结果只需要去调用一下这个递归函数就可以得到了。

递归函数的出口就是当其中一个链表为空时,直接返回另一个链表,两个都为空时,返回谁都一样。

代码:

cpp 复制代码
/**
 * Definition for singly-linked list.
 * struct ListNode {
 *     int val;
 *     ListNode *next;
 *     ListNode() : val(0), next(nullptr) {}
 *     ListNode(int x) : val(x), next(nullptr) {}
 *     ListNode(int x, ListNode *next) : val(x), next(next) {}
 * };
 */
class Solution 
{
public:
    ListNode* mergeTwoLists(ListNode* list1, ListNode* list2) 
    {
        return dfs(list1, list2);    
    }

    ListNode* dfs(ListNode* list1, ListNode* list2)
    {
        if(list1 == NULL)
            return list2;
        if(list2 == NULL)
            return list1;

        if(list1->val < list2->val)
        {
            list1->next = dfs(list1->next, list2);
            return list1;
        }
        else
        {
            list2->next = dfs(list1, list2->next);
            return list2;
        }
    }
};

反转链表

**思路:**要逆序第一个节点,我们可以先逆序第一个节点后面的子链表,逆序好后第一个节点尾插到这个子链表中即可,而逆序第一个节点后面的子链表,又可以先逆序第二个节点后面的子链表,然后把第二个节点尾插到逆序好的链表中,以此类推,这个过程就是递归的过程。

递归出口就是当遇到最后一个节点的时候没有必要逆序了,因为只有这一个节点了,直接返回就行。

代码:

cpp 复制代码
/**
 * Definition for singly-linked list.
 * struct ListNode {
 *     int val;
 *     ListNode *next;
 *     ListNode() : val(0), next(nullptr) {}
 *     ListNode(int x) : val(x), next(nullptr) {}
 *     ListNode(int x, ListNode *next) : val(x), next(next) {}
 * };
 */
class Solution {
public:
    ListNode* reverseList(ListNode* head) {
        return dfs(head);
    }

    ListNode* dfs(ListNode* list){
        if(list == NULL || list->next == NULL)
            return list;

        ListNode* node = dfs(list->next);
        ListNode* cur = node;
        while(cur->next){
            cur = cur->next;
        }
        cur->next = list;
        list->next = NULL;

        return node;
    }
};

两两交换链表中的节点

**思路:**要交换前两个节点,我们可以先将后续的链表两两交换好后,再和前两个节点按照题目要求的顺序相连接,后面的节点同理,依次类推,这就是递归的过程。

递归出口就是当只剩下一个节点的时候直接返回,只剩下两个节点的时候交换一下节点的位置然后返回。

代码:

cpp 复制代码
/**
 * Definition for singly-linked list.
 * struct ListNode {
 *     int val;
 *     ListNode *next;
 *     ListNode() : val(0), next(nullptr) {}
 *     ListNode(int x) : val(x), next(nullptr) {}
 *     ListNode(int x, ListNode *next) : val(x), next(next) {}
 * };
 */
class Solution {
public:
    ListNode* swapPairs(ListNode* head) {
        return dfs(head);
    }

    ListNode* dfs(ListNode* list){
        if(list == NULL || list->next == NULL){
            return list;
        }
        if(list->next->next == NULL){
            ListNode* head = list->next;
            list->next->next = list;
            list->next = NULL;
            return head;
        }

        ListNode* node = dfs(list->next->next);
        ListNode* head = list->next;
        list->next->next = list;
        list->next = node;

        return head;
    }
};

Pow(x, n)

**思路:**计算过程如下图:

细节问题:

  • 这道题 n 可能是负数,如果是负数可以先按正数算,然后再然 1 除以这个数就可以了。
  • 当 n 是负数最小值,转换成正数会溢出,所以要强转成 long long。

代码:

cpp 复制代码
class Solution 
{
public:
    double myPow(double x, int n) 
    {
        return n < 0 ? 1.0 / mypow(x, -(long long)n) : mypow(x, n);
    }

    double mypow(double x, long long n)
    {
        if(n == 0)
            return 1.0;

        double ret = mypow(x, n / 2);

        return n % 2 == 0 ? ret * ret : ret * ret * x;
    }
};
相关推荐
ZHOUPUYU2 小时前
PHP 8.3网关优化:我用JIT将QPS提升300%的真实踩坑录
开发语言·php
寻寻觅觅☆6 小时前
东华OJ-基础题-106-大整数相加(C++)
开发语言·c++·算法
时代的凡人6 小时前
0208晨间笔记
笔记
fpcc6 小时前
并行编程实战——CUDA编程的Parallel Task类型
c++·cuda
偷吃的耗子7 小时前
【CNN算法理解】:三、AlexNet 训练模块(附代码)
深度学习·算法·cnn
l1t7 小时前
在wsl的python 3.14.3容器中使用databend包
开发语言·数据库·python·databend
今天只学一颗糖7 小时前
1、《深入理解计算机系统》--计算机系统介绍
linux·笔记·学习·系统架构
赶路人儿7 小时前
Jsoniter(java版本)使用介绍
java·开发语言
化学在逃硬闯CS7 小时前
Leetcode1382. 将二叉搜索树变平衡
数据结构·算法
ceclar1238 小时前
C++使用format
开发语言·c++·算法