数字/字符串排列组合(Leetcode) 总结

综述:

使用递归求解问题有时往往令人费解,博主对递归也是头痛不已,以下问题,利用递归很容易求解。总的来说,递归需要设计成:处理单个问题,递归求解子问题,设置出口。需要铭记的是递归所做的工作和处理单个问题一样,所以只要单个问题取遍所有情况,那么递归同样也能取得所有情况。切记不要去想递归到底怎么一层层的调用,不管是读代码还是写代码,需要关注的是递归出口,程序怎么设计求解单个问题,一旦单个问题解决了,后续子问题就和解决单个问题一样。

1.数组的全部子集

输入:[1,2,3]

输出:[[3],[1],[2],[1,2,3],[1,3],[2,3],[1,2],[]]

这个题目如何设计递归?我们可以把输入数据分为第一个元素和剩下的元素(这样分析正符合递归的思想),那么输入的子集中只有两种情况:

1.包含第一个元素

2.不包含第一个元素

对应着程序就是取第一个元素,不取第一个元素。那么程序该如何设计?

我们可以想象:

取第一个元素,相当于将第一个元素加入到后续递归的问题中,即后续递归解中一定包含第一个元素;

不取第一个元素,那其实更简单,只需要递归求解剩下的所有元素,即略过第一个元素。

那么我们这样就已经取遍所有情况了吗?我说是的,读者不明白可以再多思考一下。别人讲明白和自己想明白完全不是一回事,后者往往更重要。

这里直接贴出Leetcode中的递归方案,并加以解释:

class Solution {
public:
    vector<vector<int>> subsets(vector<int>& nums) {
        //这里排序并不必须,只是为了子集中元素以递增排序
        sort(nums.begin(), nums.end());
        vector<vector<int>> subs;
        vector<int> sub;  
        genSubsets(nums, 0, sub, subs);
        return subs; 
    }
    void genSubsets(vector<int>& nums, int start, vector<int>& sub, vector<vector<int>>& subs) {
        //我们发现这里好像并没有递归出口,因为终止条件由下面for循环控制
        subs.push_back(sub);        
        for (int i = start; i < nums.size(); i++) {
            //情况一:子集包含元素nums[i]
            sub.push_back(nums[i]);
            //为何是i+1,因为已经解决了第i个元素,需要递归从第i+1个元素开始求解
            genSubsets(nums, i + 1, sub, subs);
            //情况二:子集不包含nums[i],即略过第i个元素
            //可以想象,不去管上一条递归语句,当下一次循环到i+1时,第i个元素已经略过
            sub.pop_back();
        }
    }
};

2.数组的全部子集(包含重复元素)

输入:[1,2,2]

输出:[[1],[2],[1,2,2],[2,2],[1,2],[]]

这和上面的问题只多了重复的元素,在上面for循环的开始加上一句:

if(i > start && nums[i] == nums[i-1]) continue;

并且,对输入数组先排序,这里就是必须的了,为何加上这个就能避免重复?一定要特别关注i>start,这说明了如果去重,必须第i-1个元素也在当前循环中,意思是for循环已经处理完第i-1个元素了,这时for循环里对i-1的递归也结束了,最后一条语句sub.pop_back()也执行完了,现在for循环要处理第i个元素了。那这时为什么要略过第i个元素呢?举个例子,因为start<=i-1,我们关注从start开始的循环,那么当循环到第i-1个元素结束时,子集中一定包含{nums[start],……,nums[i-1]},现在请注意,循环第i-1个元素结束时,第i-1个元素已经弹出,下一次循环到第i个元素时,如果nums[i]==nums[i-1]我们不略过第i元素,那么必定会产生{nums[start],……,nums[i]},这里面一定不包括nums[i-1](已经弹出),这样就产生了重复。

读者可能会问,那么像{1,2,2}这种时怎么产生的呢?这个问题留给读者自己思考。

3.数组全排列问题

输入:[1,2,2]

输出:[[1,2,3],[1,3,2],[2,1,3],[2,3,1],[3,1,2],[3,2,1]]

这是一个非常典型的递归问题,关键是如何将它转化为一个递归问题。对于单个问题,如何能够取得它的每一种情况?只有在单个问题中考虑所有的情况,自然递归子问题也能取遍所有情况。借图说明问题

 

我们还可以将输入数组看成两部分,就是第一个元素和剩余元素,我经常强调这样考虑问题的重要性。看出解决这个问题分两步:

1.对于排列,与其说第一个元素,倒不如说第一个位置。第一步就是在第一个位置取遍所有的元素

2.第二步就是按照第一步处理方式递归求解剩下的元素。

怎样才能在第一个位置取遍所有元素?我们可以将第一个元素每一次都和后面的元素交换位置,那么第一个位置就取遍了所有的元素(对应图片中的(1)->(2)),既然第一个位置所有解都已经解决,我们就求解剩下的位置,即递归求解和第一个元素交换后的剩下的元素,处理方式当然和处理第一个元素一样。第一个元素和后面元素每次交换后,每次后面的元素不可能相同,因此不用考虑递归时会出现重复的情况。结合下面代码,可能更好理解。

class Solution{
public:
	vector<vector<int>> permute(vector<int>& nums) {
		//sort(nums.begin(), nums.end());
		vector<vector<int>> subs;
		vector<int> sub;
		genSubsets(nums, 0, subs);
		return subs;
	}
	void genSubsets(vector<int>& nums, int start, vector<vector<int>>& subs) {
		if (start >= nums.size()){
			subs.push_back(nums);
			return;
		}
		for (int i = start; i < nums.size(); i++) {
			swap(nums[start], nums[i]);
			//sub.push_back(nums[i]);
			genSubsets(nums, start + 1, subs);
			swap(nums[start], nums[i]);
			//sub.pop_back();
		}
	}
};

4.数据全排列问题(包含重复元素)

输入:[1,1,2]

输出:[[1,1,2],[1,2,1],[2,1,1]]

这个问题我的解决方法是在问题3的基础上加上重复判断,如果已经包含某个子集,直接略过。

当然还有更好的方法,待完成。

5.数组子集的和为指定元素的全部组合(数组元素可重复使用)

For example, given candidate set [2, 3, 6, 7] and target 7
A solution set is: 

[
  [7],
  [2, 2, 3]
]

同样也是类似的递归方法

6.数组子集的和为指定元素的全部组合(数组元素不可重复使用)

For example, given candidate set [10, 1, 2, 7, 6, 1, 5] and target 8
A solution set is: 

[
  [1, 7],
  [1, 2, 5],
  [2, 6],
  [1, 1, 6]
]

和5稍有不同,递归时需要从下一个元素遍历。

7.回文分区问题

给出一个字符串,找出其所有的回文分割情况

For example, given s = "aab",
Return

[
  ["aa","b"],
  ["a","a","b"]
]

同样也是类似的递归求解,不同的是需要判断从start到i是否为回文。

总结:

以上可归为一类递归问题,思路相同,变化在于不同的要求,之所以后续没有给出详细解答,是因为希望学习能够举一反三,自己想明白和别人教明白是完全两回事。

参考资料:

字符串全排列

Leetcode

 欢迎大家扫描关注公众号:编程真相,获取更多精彩的编程技术文章!

  • 8
    点赞
  • 13
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
LeetCode解题总结 1. 数组 1.1 从有序数组中删除重复元素 1.2 在排序数组被旋转后进行查找 1.3 寻找两个排序数组的中位数 1.4 最长连续序列 1.5 累加和 1.6 移除数组中指定值 1.7 下一个排列 1.8 第n个全排列 1.9 验证数独的正确性 1.10 容纳雨水的量 1.11 旋转图像 1.12 数字加1 1.13 爬楼梯 1.14 格雷码 1.15 设置矩阵的行列为0 1.16 加油站问题 1.17 分糖果 1.18 只出现一次的数 2. 单链表 2.1 单链表相加 2.2 指定位置反转单链表 2.3 依据给定值将链表重新排序 2.4 删除链表中重复元素 2.5 指定位置旋转链表 2.6 删除倒数第N个节点 2.7 成对交换链表元素 2.8 复制复杂链表 2.9 链表环相关问题 2.9.1 链表是否有环 2.9.2 链表环的入口 2.10 改变链表中的元素位置2.11 LRU Cache(设计题) 3. 字符串 3.1 判断字符串是否为回文 3.2 实现strStr() 3.3 字符串转为int(atoi) 3.4 二进制树相加 3.5 最长回文字符串 3.6 正则表达式匹配[hard] 3.7 正则匹配 3.8 最长公共前缀 3.9 验证字符串是否为数字 3.10 数字转为罗马数字 3.11 罗马数字数字 3.12 Count and Say 3.13 变位词 3.14 简化系统路径 3.15 最后一个单词的长度 3.16 反转字符串中的单词 3.16.1 字符串前后和中间可能存在多个空格 3.16.2 不存在前后和中间的多余空格 3.17 一个编辑距离 4. 栈 4.1 验证括号的正确性 4.2 最长的正确括号表达式 4.3 柱状图中的最大矩形面积 4.4 计算逆波兰式的值 5. 树 5.1 二叉树的遍历 5.1.1 二叉树的前、中、后序遍历 5.1.2 二叉树的层序遍历 5.1.3 恢复二叉树[hard] 5.1.4 判断两棵树是否相等 5.1.5 判断二叉树是否为AVL树 5.1.6 将二叉树转为链表 5.1.7 二叉树添加指向右边节点的指针 5.1.8 树中节点的最小公共祖先 5.2 二叉树的构建5.3 二叉查找树 5.3.1 生成不重复的二叉查找树数目 5.3.2 验证是否为二叉查找树 5.3.3 将有序数组转为二叉树 5.3.4 将有序链表转为二叉树 5.4 二叉树的递归 5.4.1 二叉树的最大深度 5.4.2 二叉树的最小深度 5.4.3 路径和 5.4.4 满二叉树添加指向右边节点的指针 5.4.5 根节点到叶结点的所有路径代表的数字之和 6. 排序 6.1 合并两个有序数组到其中一个数组 6.2 合并两个有序链表 6.3 合并K个有序链表 6.4 使用插入排序来排序链表 6.5 归并排序排序链表 6.6 第一个缺少的正数 6.7 排序颜色 7. 查找 7.1 在排序数组中查找数出现的范围 7.2 在排序数组中查找给定值的插入位置 7.3 在二维排序数组中查找给定值 7.4 在旋转有序数组中查找最小值 7.4.1 数组无重复 7.4.2 数组有重复 7.5 在旋转排序数组中查找指定数字 8. 暴力枚举法 8.1 求集合的子集 8.2 集合的全排列 8.3 在指定树中选择进行全排列 8.4 电话上对应数字的字母组成的所有单词 9. 广度优先搜索 9.1 单词变换路径(Word Ladder) 9.1.1 是否存在变换路径 9.1.2 所有最短变换路径9.2 包围区域 10. 深度优先搜索 10.1 N皇后问题 10.2 恢复IP地址 10.3 集合元素之和 10.3.1 元素可以重复 10.3.2 元素不可重复 10.3.3 给定元素数目和元素范围 10.4 正确的括号对 10.5 解数独 10.6 单词搜索 10.7 小结 10.7.1 适用场景 10.7.2 思考步骤 10.7.3 代码模板 10.7.4 深搜与回溯、递归的区别 11. 分治法 11.1 实现pow(x, n) 11.2 Sqrt(x) 12. 贪心算法 12.1 跳台阶游戏 12.2 买卖股票的最佳时机 12.2.1 最多允许交易一次 12.2.2 可以交易任意多次 12.2.3 最多可以交易两次 12.2.4 可以交易任意多次 12.2.5 交易后需要停止一段时间 12.3 最长不含重复元素的子串 12.4 存放的最大水量 13. 动态规划 13.1 三角形从顶到底的最小路径和 13.2 最大连续子数组 13.3 字符串的所有子回文字符串 13.4 最长公共子序列问题 13.5 字符串的编辑距离 13.6 不同路径之和 13.6.1 无障碍13.6.2 有障碍 13.7 最大矩形面积 13.8 字符串交叉组合 13.9 旋转字符串 13.10 最小路径和 13.11 所有的编码方式 13.12 独一无二的子序列数 13.13 拆分单词 13.13.1 单词是否由词典中的单词组成 13.13.2 返回所有可以切分的解 14. 图 14.1 图的克隆 15. 细节实现题 15.1 反转整数 15.2 对称数判断 15.3 区间的相关操作 15.3.1 在区间中插入新的区间 15.3.2 合并区间 15.4 包含子串元素的最小窗口 15.5 大数乘法 15.6 给定串中是否存在包含所有单词的子串 15.7 Pascal 三角形 15.7.1 生成Pascal三角形 15.7.2 Pascal三角形的第N行 15.8 螺旋形矩阵 15.8.1 螺旋打印矩阵 15.8.2 生成螺旋矩阵 15.9 Z字形输出字符串 15.10 不使用乘、除、取模实现两个整数相除 15.11 文本对齐 15.12 共线的最大点数 16 其他问题 16.1 随机数生成器

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值