在解决⼀个规模为 n 的问题时,如果满足以下条件,我们可以使用递归来解决:
题目链接 -> Leetcode 面试题 08.06.汉诺塔问题
Leetcode 面试题 08.06.汉诺塔问题
题目:在经典汉诺塔问题中,有 3 根柱子及 N 个不同大小的穿孔圆盘,盘子可以滑入任意一根柱子。一开始,所有盘子自上而下按升序依次套在第一根柱子上(即每一个盘子只能放在更大的盘子上面)。移动圆盘时受到以下限制:
(1) 每次只能移动一个盘子;
(2) 盘子只能从柱子顶端滑出移到下一根柱子;
(3) 盘子只能叠在比它大的盘子上。
请编写程序,用栈将所有盘子从第一根柱子移到最后一根柱子。
你需要原地修改栈。
示例1 :
输入:A = [2, 1, 0], B = [], C = []
输出:C = [2, 1, 0]
示例2 :
输入:A = [1, 0], B = [], C = []
输出:C = [1, 0]
提示 :
思路:
这是一道递归方法的经典题目,我们可以先从最简单的情况考虑:
因为 A 中最后处理的是最大的盘子,所以在移动过程中不存在大盘子在小盘子上面的情况。
则本题可以被解释为:
代码如下:
class Solution {
public:
void hanota(vector& A, vector& B, vector& C)
{
// 将 A 上的 A.size() 个盘子借助 B 移到 C 上
dfs(A, B, C, A.size());
}
void dfs(vector& A, vector& B, vector& C, int n)
{
// 递归出口
if(n == 1)
{
C.push_back(A.back());
A.pop_back();
return;
}
// 将 A 上的 n - 1 个盘子借助 C 移到 B 上
dfs(A, C, B, n - 1);
C.push_back(A.back());
A.pop_back();
// 将 B 上的 n - 1 个盘子借助 A 移到 C 上
dfs(B, A, C, n - 1);
}
};
题目链接 - > Leetcode -21.合并两个有序链表
Leetcode -21.合并两个有序链表
题目:将两个升序链表合并为一个新的 升序 链表并返回。新链表是通过拼接给定的两个链表的所有节点组成的。
示例 1:
输入:l1 = [1, 2, 4], l2 = [1, 3, 4]
输出:[1, 1, 2, 3, 4, 4]
示例 2:
输入:l1 = [], l2 = []
输出:[]
示例 3:
输入:l1 = [], l2 = [0]
输出:[0]
提示:
思路:
代码如下:
class Solution
{
public:
ListNode* mergeTwoLists(ListNode* list1, ListNode* list2)
{
if(list1 == nullptr) return list2;
if(list2 == nullptr) return list1;
if(list1->val < list2->val)
{
list1->next = mergeTwoLists(list1->next, list2);
return list1;
}
else
{
list2->next = mergeTwoLists(list1, list2->next);
return list2;
}
}
};
题目链接 -> Leetcode -206.反转链表
Leetcode -206.反转链表
题目:给你单链表的头节点 head ,请你反转链表,并返回反转后的链表。
示例 1:
输入:head = [1, 2, 3, 4, 5]
输出:[5, 4, 3, 2, 1]
示例 2:
输入:head = [1, 2]
输出:[2, 1]
示例 3:
输入:head = []
输出:[]
提示:
思路:
代码如下:
/**
* 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)
{
if(head == nullptr || head->next == nullptr) return head;
ListNode* ret = reverseList(head->next);
head->next->next = head;
head->next = nullptr;
return ret;
}
};
题目链接 -> Leetcode -24.两两交换链表中的节点
Leetcode -24.两两交换链表中的节点
题目:给你一个链表,两两交换其中相邻的节点,并返回交换后链表的头节点。
你必须在不修改节点内部的值的情况下完成本题(即,只能进行节点交换)。
示例 1:
输入:head = [1, 2, 3, 4]
输出:[2, 1, 4, 3]
示例 2:
输入:head = []
输出:[]
示例 3:
输入:head = [1]
输出:[1]
提示:
思路:
代码如下:
/**
* 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)
{
if (head == nullptr || head->next == nullptr) return head;
ListNode* tmp = swapPairs(head->next->next);
ListNode* ret = head->next; // 先存一下返回的头节点
head->next->next = head;
head->next = tmp;
return ret;
}
};
题目链接 -> Leetcode -50.Pow(x, n)
Leetcode -50.Pow(x, n)
题目:实现 pow(x, n) ,即计算 x 的整数 n 次幂函数(即,xn )。
示例 1:
输入:x = 2.00000, n = 10
输出:1024.00000
示例 2:
输入:x = 2.10000, n = 3
输出:9.26100
示例 3:
输入:x = 2.00000, n = -2
输出:0.25000
解释:2 - 2 = 1 / 22 = 1 / 4 = 0.25
提示:
思路:
代码如下:
class Solution {
public:
// 2^10 = 2^5 * 2^5 = (2^2 * 2^2 * 2) * (2^2 * 2^2 * 2) = ...
double _pow(double x, int n)
{
if(n == 0) return 1.0;
double tmp = _pow(x, n / 2);
return n % 2 == 0? tmp * tmp : tmp * tmp * x;
}
double myPow(double x, int n)
{
return n >= 0? _pow(x, n) : 1.0 / _pow(x, abs(n));
}
};