全排列算法(一文看懂全排列算法)
全排列算法(一文看懂全排列算法)
所谓全排列,就是把一堆字符按照一定的顺序排列起来,所有可能的组合。
举个简单的例子,"123"的全排列为"123"、"132"、"213"、"231"、"312"、"321"。
使用库函数进行全排列
C++的<algorithm>头文件中实现了全排列,即next_permutation函数,它是基于字典序实现的,执行一次next_permutation函数就相当于进行了一次"变异",变异之后字典序会比原来的字符串大,但其位次也仅仅排在变异之前的字符串之后。什么意思呢?比如"123"调用next_permutation函数经过一次变异之后会变成"132",而不是"213"、"321"等字典序更大的字符串。
next_permutation是有返回值的,返回值为true或false,意思是如果变异之后仍然产生了新的排列就会返回true,不能再产生新的排列了就会返回false。还是上面那个例子,如果当前字符串已经是"321"了,不存在字典序比"321"更大的排列了,此时就会返回false。因此,如果要进行全排列的字符串是"132",就应当先对其排序变成"123",否则在全排列时就会漏掉"123"。
next_permutation的用法如下:#include<iostream>#include<algorithm>usingnamespacestd;stringstr;intmain{getline(cin,str);//先进行排序使之字典序最小sort(str.begin,str.end);cout<<"其全排列为:"<<endl;do{cout<<str<<endl;}while(next_permutation(str.begin,str.end));return0;}
手撕全排列
可是如何用编程实现这一过程呢?本文就教大家使用深搜回溯算法实现全排列。代码如下:#include<iostream> #include<vector>#include<algorithm>usingnamespacestd;stringstr;stringtemp;vector<bool>vis;voiddfs(intcnt,intn){if(cnt==n){cout<<temp<<endl;return;}for(inti=0;i<n;i++){if(vis[i]==false){temp.push_back(str[i]);vis[i]=true;dfs(cnt+1,n);vis[i]=false;temp.pop_back;}}}intmain{getline(cin,str);sort(str.begin,str.end);intn=str.size;vis.resize(n);dfs(0,n);return0;}
我们把产生的排练暂存在字符串temp中,当temp中的字符个数与初始字符串的字符个数相等时就将temp输出了。
数组vis存放的是字符串的某个下标索引到的字符有没有加入temp,如果加入了temp就将其vis置为true,没加入的其vis就为false。
dfs传入的参数cnt是指已经填入temp的字符个数,n是初始字符串的字符个数。
有了上面那些铺垫,我们在主函数中调用dfs(0,n),意思是初始状态temp中没有字符。
为了建立字符与下标之间的联系,方便大家观察,我们使用"012"这个例子描述算法,最初temp={},vis都为false,进了递归函数dfs,就开始遍历初始字符串,依次往temp填入字符。
在阅读下面的过程之前,我邀请大家关注两个要素,递归层数和当前递归层数下i的值,这两个要素直接决定了下一个要尝试填入的字符。
起初递归层数为0。从i=0开始遍历。i=0时,vis[0]=false,将0填入temp,然后将vis[0]置为true,传入cnt+1=1表示temp中已有字符的个数为1,进行下一层递归,此时
temp={0}
此时递归层数为1。从i=0开始遍历。i=0时,vis[0]=true,0已经被填入temp了不满足条件;i=1时,vis[1]=false,将1填入temp,然后将vis[1]置为true,传入cnt+1=2表示temp中已有字符的个数为2,进行下一层递归,此时
temp={0,1}
此时递归层数为2。从i=0开始遍历。i=0时,vis[0]=true,0已经被填入temp了不满足条件;i=1时,vis[1]=true,1已经被填入temp了不满足条件;i=2时,vis[2]=false,将2填入temp,然后将vis[2]置为true,传入cnt+1=3表示temp中已有字符的个数为3,进行下一层递归,此时
temp={0,1,2}
此时递归层数为3。经判断temp中已经填入了3个字符,达到了"出厂要求",将其输出后返回上一层递归。
此时递归层数为2。上次在2层递归里遍历到i=2了,从dfs返回后取出temp末尾的字符2,于是将vis[2]又置为了false,继续遍历,i=3超出了下标范围,遍历结束,返回上一层递归。此时
temp={0,1}
此时递归层数为1。上次在1层递归里遍历到i=1了,从dfs返回后取出temp末尾的字符1,于是将vis[1]又置为了false;此时
temp={0}
继续遍历,此时i=2,vis[2]=false,将2填入temp,然后将vis[2]置为true,传入cnt+1=2表示temp中已有字符的个数为2,进行下一层递归,此时
temp={0,2}
此时递归层数为2。i=0时,vis[0]=true,0已经被填入temp了不满足条件;i=1时,vis[1]=false,将1填入temp,然后将vis[1]置为true,传入cnt+1=3表示temp中已有字符的个数为3,进行下一层递归,此时
temp={0,2,1}
此时递归层数为3。经判断temp中已经填入了3个字符,达到了"出厂要求",将其输出后返回上一层递归。
此时递归层数为2。上次在2层递归里遍历到i=1了,从dfs返回后取出temp末尾的字符1,于是将vis[1]又置为了false。此时
temp={0,2}
继续遍历,此时i=2,vis[2]=true,2已经被填入temp了不满足条件;继续遍历,i=3超出了下标范围,遍历结束,返回上一层递归。
此时递归层数为1。上次在1层递归里遍历到i=2了,从dfs返回后取出temp末尾的字符2,于是将vis[2]又置为了false。此时
temp={0}
继续遍历,i=3超出了下标范围,遍历结束,返回上一层递归。
此时递归层数为0。上次在0层递归里遍历到i=0了,从dfs返回后取出temp末尾的字符0,于是将vis[0]又置为了false。此时
temp={}
继续遍历,此时i=1,vis[1]=false,将1填入temp,并将vis[1]置为true,传入cnt+1=1表示temp中已有字符的个数为1,进行下一层递归,此时
temp={1}
此时递归层数为1。从i=0开始遍历。i=0时,vis[0]=false,将0填入temp,然后将vis[0]置为true,传入cnt+1=2表示temp中已有字符的个数为2,进行下一层递归,此时
temp={1,0}
此时递归层数为2。从i=0开始遍历。i=0时,vis[0]=true,0已经被填入temp了不满足条件;i=1时,vis[1]=true,1已经被填入temp了不满足条件;i=2时,vis[2]=false,将2填入temp,然后将vis[2]置为true,传入cnt+1=3表示temp中已有字符的个数为3,进行下一层递归,此时
temp={1,0,2}
此时递归层数为3。经判断temp中已经填入了3个字符,达到了"出厂要求",将其输出后返回上一层递归。
此时递归层数为2。上次在2层递归里遍历到i=2了,从dfs返回后取出temp末尾的字符2,于是将vis[2]又置为了false;继续遍历,i=3超出了下标范围,遍历结束,返回上一层递归。此时
temp={1,0}
此时递归层数为1。上次在1层递归里遍历到i=0了,从dfs返回后取出temp末尾的字符0,于是将vis[0]又置为了false;此时
temp={1}
继续遍历,此时i=1,vis[1]=true,1已经被填入temp了不满足条件;继续遍历,此时i=2,vis[2]=false,将2填入temp,然后将vis[2]置为true,传入cnt+1=2表示temp中已有字符的个数为2,进行下一层递归,此时
temp={1,2}
此时递归层数为2。从i=0开始遍历。i=0时,vis[0]=false,将0填入temp,然后将vis[0]置为true,传入cnt+1=3表示temp中已有字符的个数为3,进行下一层递归,此时
temp={1,2,0}
此时递归层数为3。经判断temp中已经填入了3个字符,达到了"出厂要求",将其输出后返回上一层递归。
此时递归层数为2。上次在2层递归里遍历到i=0了,从dfs返回后取出temp末尾的字符,其实就是0,于是将vis[0]又置为了false;继续遍历,vis[1]和vis[2]都为true,也就是1和2都在temp里,都不满足条件,遍历结束返回上一层递归。此时
temp={1,2}
此时递归层数为1。上次在1层递归里遍历到i=2了,从dfs返回后取出temp末尾的字符2,于是将vis[2]又置为了false;此时
temp={1}
继续遍历,i=3超出了下标范围,遍历结束,返回上一层递归。此时
此时递归层数为0。上次在0层递归里遍历到i=1了,从dfs返回后取出temp末尾的字符1,于是将vis[1]又置为了false;此时
temp={}
继续遍历,此时i=2,vis[2]=false,将2填入temp,然后将vis[2]置为true,传入cnt+1=1表示temp中已有字符的个数为1,进行下一层递归,此时
temp={2}
此时递归层数为1。从i=0开始遍历。i=0时,vis[0]=false,将0填入temp,然后将vis[0]置为true,传入cnt+1=2表示temp中已有字符的个数为2,进行下一层递归,此时
temp={2,0}
此时递归层数为2。从i=0开始遍历。i=0时,vis[0]=true,0已经被填入temp了不满足条件;i=1时,vis[1]=false,将1填入temp,然后将vis[1]置为true,传入cnt+1=3表示temp中已有字符的个数为3,进行下一层递归,此时
temp={2,0,1}
此时递归层数为3。经判断temp中已经填入了3个字符,达到了"出厂要求",将其输出后返回上一层递归。
此时递归层数为2。上次在2层递归里遍历到i=1了,从dfs返回后取出temp末尾的字符1,于是将vis[1]又置为了false;此时
temp={2,0}
继续遍历,此时i=2,vis[2]=true,2已经被填入temp了不满足条件;继续遍历,i=3超出了下标范围,遍历结束,返回上一层递归。
此时递归层数为1。上次在1层递归里遍历到i=0了,从dfs返回后取出temp末尾的字符0,于是将vis[0]又置为了false;此时
temp={2}
继续遍历,此时i=1,vis[1]=false,将1填入temp,然后将vis[1]置为true,传入cnt+1=2表示temp中已有字符的个数为2,进行下一层递归,此时
temp={2,1}
此时递归层数为2。从i=0开始遍历。i=0时,vis[0]=false,将0填入temp,然后将vis[0]置为true,传入cnt+1=3表示temp中已有字符的个数为3,进行下一层递归,此时
temp={2,1,0}
此时递归层数为3。经判断temp中已经填入了3个字符,达到了"出厂要求",将其输出后返回上一层递归。
此时递归层数为2。上次在2层递归里遍历到i=0了,从dfs返回后取出temp末尾的字符0,于是将vis[0]又置为了false;此时
temp={2,1}
继续遍历,vis[1]和vis[2]都为true,意味着1和2都在temp里,都不满足条件,遍历结束,返回上一层递归。
此时递归层数为1。上次在1层递归里遍历到i=1了,从dfs返回后取出temp末尾的字符1,于是将vis[1]又置为了false;此时
temp={2}
继续遍历,此时i=2,vis[2]=true,2已经被填入temp了不满足条件;继续遍历,i=3超出了下标范围,遍历结束,返回上一层递归。
此时递归层数为0。上次在0层递归里遍历到i=2了,从dfs返回后取出temp末尾的字符2,于是将vis[2]又置为了false。此时
temp={}
继续遍历,i=3超出了下标范围,遍历结束。
全排列到此结束,temp和vis都恢复了最初的状态。
又到了金三银四面试季,衷心祝愿大家都能拿到想要的Offer。
公积金一个月多少(公积金交多少钱一个月)公积金知识小科普您每个月缴存多少公积金?住房公积金缴存基数为职工本人上一年度月平均工资。单位每年应当根据职工上一年度工资调整情况确定当年的住房公积金缴存基数。目前住房公积金缴存比例
金牛角王(长沙金牛角王为什么停业)金牛角王(长沙金牛角王为什么停业)金牛角王停业,没能挺过牛年立春日在湘7家门店全部关闭,充值卡金额启动债权登记长沙的初代网红,80后满满都是回忆2月3日,前一日正常营业的金牛角王富
技师怎么样(做技师以后会是什么样的人)今天给大家讲讲我当技师那些年的事。随着近几年健康行业的普及,人们对养生的这个话题比较。所谓技师很多局外的人认为不就是给人按按摩做个足疗,厉害点的卡巴卡巴掰几下,正个骨啥的。其实,根
为什么不建议吃沐舒坦(氨溴索为什么越用越咳)为什么不建议吃沐舒坦(氨溴索为什么越用越咳)这两天真的太太太太太冷了!要问这个时候,老父亲老母亲最怕的是什么?那当然是宝宝生病咳嗽了!一听到娃咳咳咳,爸妈这心里就像炸毛了一样,暴躁
皮包有哪些(真皮包)米娜桑,空哩即哇很多小伙伴会找日代买一些美妆护肤品家用电器啥的,但其实日本还有一个隐藏名品是值得入手的哦那就是手作皮革单品!这些手工制作的皮具们,品质完全不输大牌,甚至比一些大几万
xlsx是什么格式(xlsx是什么文件)xlsx办公软件电脑版是由微软公司推出的一个OfficeExcel扩展名文档软件,它的功能十分强大,可以帮助用户快速的进行处理需要的数据,支持更好的进行格式化分析呈现出数据等,并且
超市促销大妈爬上货架哄抢近日超市促销大妈爬上货架哄抢登录了热搜,也是在网上引起了网友们的关注,那么很多小伙伴可能还不清楚具体的情况如何,小编也是在网上查阅了一些信息,那么接下来就分享给大家来了解下超市促销
今日应县高梁价格,应县官网平时妈妈说的全部都是骗人的,挺轻松的,房地产项目开盘时间,应县房价,特惠团购山西朔州应县海景房5千以,涂擦SPF大于1PA防晒护肤品,为您提供应县今年车险费用怎么交怎么买,户型图片
电脑电源不够会怎么样(电源超负荷的后果是什么)在如今电脑对于我们的生活来说已经成为了不可或缺的一部分了,相信我们大家许多人在使用的过程中多多少少都会碰到一些问题。电源作为电脑动力的来源是非常重要的,直接影响着电脑硬件的稳定运行
电源功率不够会怎么样(电源电流不够会怎么样)谣零零计划近期,出现了不少RTX3090炸卡的现象,这真是黄老板翻车了吗?消费者反映,这些失灵的3090显卡的故障几乎都相同,在正常使用的时候,突然画面没有输出,同时闻到了刺鼻的糊
什么是开关电源(开关电源的分类方法)一开关电源是一种高频化电能转换装置,其主要利用电力电子开关器件(如晶体管MOS管可控晶闸管等),通过控制电路,使电子开关器件周期性地接通和关断,让电力电子开关器件对输入电压进行脉冲
95后博士连续两年获全球数学金奖近日有关于95后博士连续两年获全球数学金奖的问题受到了很多网友们的关注,大多数网友都想要知道95后博士连续两年获全球数学金奖的具体情况,那么关于到95后博士连续两年获全球数学金奖的
LUXGEN5Sedan挑战台湾精品金质奖台湾精品谁最讚!第21届台湾金质奖大预测活动热烈展开LUXGEN汽车以LUXGEN5Sedan第四度蝉联台湾精品奖殊荣,昨日(4月2日)中华民国外贸协会特地假台北国际贸易中心举办2
BMW大5系列获颁2018车讯风云奖最佳进口大型车殊荣迈入第十三届的2018车讯风云奖评选结果于4月13日正式公布,通过38位来自汽车媒体各领域专业人士及消费者代表的严谨评选,经过二轮的不记名投票决选,BMW大5系列最终荣获多数评审的
员工福利好工作效率高中华汽车获颁康健杂誌乐活企业奖近年全球吹起一阵乐活旋风,为顺应这风潮,康健杂誌特举办乐活企业调查中华汽车从缜密的调查评比中脱颖而出,荣获2008乐活企业奖,3月5日当天,由中华汽车陈茂顺副总代表出席授奖,并分享
张玉环称曾遭6天6夜刑讯逼供希望还一个公道近日张玉环称曾遭6天6夜刑讯逼供登录了热搜,也是在网上引起了网友们的关注,那么很多小伙伴可能还不清楚具体的情况如何,小编也是在网上查阅了一些信息,那么接下来就分享给大家来了解下张玉
广州桥中房业主称想尽快还路于民强调自己只需要一套合近日,广州市环岛路海珠涌大桥下的一户ldquo桥中房rdquo火了,不少民众看了报道后都前往拍照打卡。对于舆论关切,业主梁小姐再次回应称,网传天价赔偿和要求置换四套房子的消息不实,
饿了么称希望把选择权交给用户近日饿了么称希望把选择权交给用户登录了热搜,也是在网上引起了网友们的关注,那么很多小伙伴可能还不清楚具体的情况如何,小编也是在网上查阅了一些信息,那么接下来就分享给大家来了解下饿了
民警查洪失联3天见到同事泪奔最近不少地区暴雨连连,截止8月9日22时洛南县暴雨洪涝灾害导致陕西洛南县5人死亡10人失联,28037户85204人受灾。而在8月5日下午,陕西商洛洛南县突降暴雨,洛南交警大队4位
伊拉克将向黎巴嫩派遣救援医疗队据消息透露,8月4日晚间,伊拉克总理卡迪米总统萨利赫及议长哈布希分别就贝鲁特发生爆炸一事声援黎巴嫩政府及人民,向事件遇难者表示哀悼,并表明伊拉克政府对黎巴嫩的坚定支持。总理卡迪米向
伊拉克将向黎巴嫩派遣救援医疗队近日伊拉克将向黎巴嫩派遣救援医疗队登录了热搜,也是在网上引起了网友们的关注,那么很多小伙伴可能还不清楚具体的情况如何,小编也是在网上查阅了一些信息,那么接下来就分享给大家来了解下伊
如何改善民生(如何保障和改善民生问题)来源人民网人民日报习近平总书记在庆祝中国共产党成立100周年大会上的重要讲话中强调坚持在发展中保障和改善民生。民生是人民幸福之基社会和谐之本,增进民生福祉是发展的根本目的。坚持在发