本文解说4道关于permutation的题目:

1. Permutation:输出permutation——基础递归 
2. Permutation Sequence: 输出字典序排列的第k个permutation——推理
3. Next Permutation:给定一个permutation中的序列,求字典序它的下一个permutation是什么——逻辑推理
4. Permutation II:和第一题有细微的区别: 对于一个可能有反复元素的数组输出全部permutation——有条件dfs

                                                                                             

1. Permutation:输出permutation

——基础递归

 class Solution{
private:
vector<vector<int> > L;
vector<int> Nums;
int l;
vector<bool> visited;
public:
void perm(vector<int>& list){
if(list.size()==l){
L.push_back(list);
return;
}
for(int i=0; i<l; i++){
if(!visited[i]){
list.push_back(Nums[i]);
visited[i] = true;
perm(list);
list.pop_back();
visited[i] = false;
}
}
} vector<vector<int> > permute(vector<int> &num){
int i;
Nums = num;
l = Nums.size();
for(i=0;i<l;i++)
visited.push_back(false);
vector<int>list;
perm(list); //for(i=0;i<L.size();i++){
// for(int j = 0;j<l;j++)
// cout<<L[i][j];
// cout<<endl;
//}
return L;
}
};

                                                                                             

2. Permutation Sequence: 输出字典序排列的第k个permutation

——逻辑推理

酱想。n位的permutation有n!个。那么第k个permutation假设满足n!<k<(n+1)!就一定有,

a = k/n!

b = k%n!

取集合里的第a位做下一位。下一次分析剩下的字符组成的第b个permutation

------------------------

e.g. 求[1,2,3,4]组合的第10个

①求[1,2,3,4]组合的第10个

10/3! = 1…4 --->找到[1,2,3,4]中第(1+1)个数(2)做下一位,留下[1,3,4]

②求[1,3, 4]组合的第10-3! * 1 = 4个

4/2! = 2…0 --->找到[1,3,4]中第2个数(3)做下一位, 留下[1,4]

③余零。说明是permutation里的最后一个 -> 剩下的逆序输出

--->2341

class Solution {
public:
string getPermutation(int n, int k) {
int i,j,sum = 1;
//sum = (n-1)!
for (i=2; i<n; i++) {
sum *= i;
}
bool visited[n+2];
memset(visited, false, sizeof(visited));
string str;
for(i=1;i<n;i++){
int nextidx = k/sum;
k = k%sum;
if(k==0)
nextidx -- ;
sum/=(n-i);
int cnt = 0;
for (j=0; j<n; j++) {
if (!visited[j]) {
if (cnt == nextidx){
visited[j] = true;
str += '0' +j+1;
break;
}
cnt ++;
}
}
}
for(j=n-1;j>=0;j--){
if (!visited[j]) {
str += '0' + j+1;
}
}
return str; }
};
                                                                                             

3. Next Permutation

给定一个permutation中的序列,求字典序它的下一个permutation是什么。

——逻辑推理

能够发现,下一个permutation能够这么得来:

①当前permutation从后往前找到一直上升的子序列,假如一直上升到index_i

②找到index为i到end中最小的。比num[i-1]大的数字,记index为j。交换num[i-1],num[j]

③对num[i]~num[end]从小到大排序

PS:要注意有反复元素的情况e.g {1,5,1};

code:

class Solution {
public:
void nextPermutation(vector<int> &num) {
size_t n = num.size();
int i = (int)n-1;
int j=0;//find the position that stops increasing from tail
while(num[i]<=num[i-1] && i>0)
i--; sort(num.begin()+i, num.end());
//find the digit that substitute(swap with) i
for(j=i;j<n;j++){
if (num[j]>num[i-1]) {
break;
}
}
if(i>0 && j<n)
swap(num[i-1], num[j]);
}
};
                                                                                             


4. Permutation II

和第一题有细微的区别: 对于一个可能有反复元素的数组输出全部permutation。

——有条件dfs

想一下递归条件:

肯定还是递归,递归条件应该是假设当前list中已经出现过这几个元素排列。就不要再加进去。

所以在第一题基础上仅仅加两点:

1)对数组里全部元素排序

2)对于上一次加到过list的同样元素(必定是在排序后数组中与上一个相邻元素同样的)不要再加

class Solution{
private:
vector<vector<int> > L;
vector<int> Nums;
int l;
vector<bool> visited;
public:
void perm(vector<int>& list){
if(list.size()==l){
L.push_back(list);
return;
}
for(int i=0; i<l; i++){
if(!visited[i]){
list.push_back(Nums[i]);
visited[i] = true;
perm(list);
list.pop_back();
visited[i] = false; while (i<l && Nums[i+1]==Nums[i]) {
i++;
}
}
}
} vector<vector<int> > permuteUnique(vector<int> &num){
int i;
Nums = num;
l = Nums.size();
sort(Nums.begin(),Nums.end());
for(i=0;i<l;i++)
visited.push_back(false);
vector<int>list;
perm(list); return L;
}
};

排列组合(permutation)系列解题报告的更多相关文章

  1. LeetCode: Permutation Sequence 解题报告

    Permutation Sequence https://oj.leetcode.com/problems/permutation-sequence/ The set [1,2,3,…,n] cont ...

  2. 【LeetCode】60. Permutation Sequence 解题报告(Python & C++)

    作者: 负雪明烛 id: fuxuemingzhu 个人博客: http://fuxuemingzhu.cn/ 目录 题目描述 题目大意 解题方法 日期 题目地址:https://leetcode.c ...

  3. SPOJ QTREE 系列解题报告

    题目一 : SPOJ 375 Query On a Tree http://www.spoj.com/problems/QTREE/ 给一个树,求a,b路径上最大边权,或者修改a,b边权为t. #in ...

  4. 自然语言处理(NLP) - 数学基础(1) - 排列组合

    正如我在<自然语言处理(NLP) - 数学基础(1) - 总述>一文中所提到的NLP所关联的概率论(Probability Theory)知识点是如此的多, 饭只能一口一口地吃了, 我们先 ...

  5. 【LeetCode】266. Palindrome Permutation 解题报告(C++)

    作者: 负雪明烛 id: fuxuemingzhu 个人博客:http://fuxuemingzhu.cn/ 目录 题目描述 题目大意 解题方法 字典 日期 题目地址:https://leetcode ...

  6. 【剑指Offer】字符串的排列 解题报告(Python)

    [剑指Offer]字符串的排列 解题报告(Python) 标签(空格分隔): LeetCode 题目地址:https://www.nowcoder.com/ta/coding-interviews 题 ...

  7. LeetCode解题报告—— Jump Game & Merge Intervals & Permutation Sequence

    1. Jump Game Given an array of non-negative integers, you are initially positioned at the first inde ...

  8. 【九度OJ】题目1072:有多少不同的面值组合? 解题报告

    [九度OJ]题目1072:有多少不同的面值组合? 解题报告 标签(空格分隔): 九度OJ http://ac.jobdu.com/problem.php?pid=1072 题目描述: 某人有8角的邮票 ...

  9. 【排列组合】ZSC1076: 数学、不容易系列之三——考新郎

    国庆期间,省城刚刚举行了一场盛大的集体婚礼,为了使婚礼进行的丰富一些,司仪临时想出了有一个有意思的节目,叫做"考新郎",具体的操作是这样的: 首先,给每位新娘打扮得几乎一模一样,并 ...

随机推荐

  1. 我用的主机,推荐给大家【gegehost】【戈戈主机】

    炎炎夏日冰点价格:戈戈主机史上最大优惠促销活动 1.7月1日至8日:买主机优惠大促销:主机买一送一,不限购买数量 请您通过客户中心或者淘宝购买一个主机之后,登录客户中心,提交问题, 提供要赠送的主机的 ...

  2. Friday Q&A 2015-11-20:协变与逆变

    作者:Mike Ash,原文链接,原文日期:2015-11-20译者:Cee:校对:千叶知风:定稿:numbbbbb 在现代的编程语言中,子类型(Subtypes)和超类型(Supertypes)已经 ...

  3. webgl推荐书籍

    网址:https://www.douban.com/doulist/45940373/ webgl 来自: Pasu2017-04-17创建   2017-07-25更新   推荐 关注 2 人关注 ...

  4. RackTables在LNMP系统的安装及使用

    RackTables是一款优秀的机房管理系统,可以十分方便的登记机房设备和连接情况,非常适合小型机房的运维.RackTables是PHP开发的免费系统,最新版本为0.20.14,PHP版本要求不低于P ...

  5. 小b和灯泡

    2489 小b和灯泡 2 秒 262,144 KB 10 分 2 级题   小b有n个关闭的灯泡,编号为1...n. 小b会进行n轮操作,第i轮她会将编号为i的倍数的灯泡的开关状态取反,即开变成关,关 ...

  6. git 提交运用vim编辑器

    git  commit -m 默认使用nano,觉得不爽,改成vim吧.在 .gitconfig (在根目录下)的  [core] 段中加上 editor=vim . 或:$git config -- ...

  7. 自媒体人Chrome浏览器必备插件精选神器!

    自从互联网时代起,浏览器使用从最早的IE,到opera,到猎豹浏览器,到360双核浏览器,到火狐,到safari,到目前最喜欢用的chrome.一路下来,chrome的稳定性与扩展性征服了我,成了我必 ...

  8. Python之游戏开发-飞机大战

    Python之游戏开发-飞机大战 想要代码文件,可以加我微信:nickchen121 #!/usr/bin/env python # coding: utf-8 import pygame impor ...

  9. Python之字符串计算(计算器)

    Python之字符串计算(计算器) import re expression = '-1-2*((60+2*(-3-40.0+42425/5)*(9-2*5/3+357/553/3*99/4*2998 ...

  10. python第一章计算机基础

    第一章 计算机基础 1.1 硬件 计算机基本的硬件由:CPU / 内存 / 主板 / 硬盘 / 网卡 / 显卡 / 显示器 等组成,只有硬件但硬件之间无法进行交流和通信. 1.2 操作系统 操作系统用 ...