教育房产时事环球科技商业
投稿投诉
商业财经
热点动态
科技数码
软件应用
国际环球
晨报科学
新闻时事
信息智能
汽车房产
办公手机
教育体育
生活生物

全排列算法(一文看懂全排列算法)

  全排列算法(一文看懂全排列算法)
  所谓全排列,就是把一堆字符按照一定的顺序排列起来,所有可能的组合。
  举个简单的例子,"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。

兔兔怎么样(兔兔解析)兔子人工养殖利用时间长,技术成熟,并不属于2020年禁养名单,因此还是可以发展人工养殖的,那么2020养兔子利润怎么样?成本需要多少?一起来了解兔子人工养殖成本利润。2020养兔子的多音字有哪些(曷多音字组词)最新部编版五(上)全册多音字觉ju感觉jio睡觉澄chng清澄dng澄沙别bi别人bi别扭散sn散文sn散步空kng天空kng空地便pin便宜bin方便盛shng盛开chng盛饭应治疗风湿最好的方法(出风湿的最好的方法)治疗风湿最好的方法(出风湿的最好的方法)现实生活当中,有很多人被类风湿这种疾病所困扰,类风湿不仅会导致关节的红肿疼痛,还会引起关节变形僵硬,甚至失去正常的行动能力。因此用什么方法治格温为什么跟绿魔睡(钢铁侠睡白皇后漫画)DC的一哥是蝙蝠侠,是不是很多人认为漫威的一哥是钢铁侠?但实际上漫威的一哥是蜘蛛侠,有漫威的大佬斯坦李和画家史蒂芬迪科特联合创作。当年斯坦李提出要创造的是一个街头英雄,本身还是个高一地两检争议是什么(争议两个小时)香港保安局局长李家超3月2日在立法会保安事务委员会会议上表示,皇岗新口岸日后将沿用目前深圳湾口岸的一地两检通关模式,特区政府将会在新皇岗口岸大楼内设立港方口岸区,并行使全面司法管辖黄渤为什么是影帝(影帝和影后是什么意思)从喜剧王中王到草根,东方卓别林黄渤的影帝之路,为何越走越退步。黄渤曾被找周星驰评价为喜剧王中王,但是事实上,黄渤刚出道的时候,却是以偶像歌手出道的,不过从歌手到百亿票房影帝,黄渤总空调滴水多久(空调制热内机滴水吗)你是否有过这样的经历宁静的夏夜,在房间里吹着空调玩手机,却突然听见,空调传来滴滴哒哒的水声其实,空调制冷时,室内的空气在温度很低的蒸发器上遇冷会产生冷凝水。正常情况下,冷凝水会顺着醉酒男子24楼抛下哑铃路人险被砸近日有关于醉酒男子24楼抛下哑铃路人险被砸的问题受到了很多网友们的关注,大多数网友都想要知道醉酒男子24楼抛下哑铃路人险被砸的具体情况,那么关于到醉酒男子24楼抛下哑铃路人险被砸的今日热点新闻醉酒男子24楼抛下哑铃路人险被砸近日有关于醉酒男子24楼抛下哑铃路人险被砸的问题受到了很多网友们的关注,大多数网友都想要知道醉酒男子24楼抛下哑铃路人险被砸的具体情况,那么关于到醉酒男子24楼抛下哑铃路人险被砸的新闻醉酒男子持刀行凶店主挺身而出近日醉酒男子持刀行凶店主挺身而出登录上了百度热搜,受到广大网友们的关注,那么关于目前的醉酒男子持刀行凶店主挺身而出相信小伙伴们都是想要了解到最新的信息吧,小编也是在网上进行了一些整鸡鸡为什么会痛(男孩小鸡疼怎么回事)之所以男性易患上生殖系统疾病,主要跟生理构造相关,特别是前列腺炎阴囊炎和睾丸炎等,不管是哪种生殖系统疾病都会增加身体不适,而且也会降低性生活质量。其中最让男性难以启齿的莫过于阴茎疼
信用卡多久提升额度(信用卡用多久能提升到100万)提高永久额度,比较特殊的是交通银行和中信银行,交通银行信用卡通常不用用户申请提额,银行每个月会对用户的用卡情况及刷卡额度等进行综合评测,然后进行自动调额。而中信银行信用卡则是无具体广发diy信用卡额度一般是多少(信用卡怎么提升额度快)2020年该如何用卡一信用体系更加完善征信是一个人信用良好与否的晴雨表,无论是贷款还是信用卡,都会参考征信报告。文章开头提到了新版征信百行征信以及大数据,而且数据更新将缩短到一个工怎么提高信用卡额度(信用卡该怎么还)信用卡提额方法汇总正常来说满足两个条件信用卡就可以提额第一轻负债第二多刷卡很多人的额度停滞都是因为负债二字,当然也有一部分老铁是因为授信到了天花板!回归正文,今天简单聊一下信用卡的信用卡额度多少(信用卡20万额度算高吗)银行信用卡的额度一般是多少?这应该是大多数人在申请信用卡之前都会考虑的问题。考虑到每位申请人的个人资质的不同,以及各家银行的审核标准的不同,信用卡最终审批的额度也不尽相同。但有一点农行哪个信用卡好(农行信用卡哪个额度高)农业银行作为四大行,发行的信用卡种类不多,没有独立的信用卡业务部门,申请信用卡,要么网申,要么网点经理,没有办卡业务员(外包)。在申请农行卡之前,有必要先研究透它的套路。一刚性减扣哪个信用卡好办理(哪个信用卡额度高)信用卡为我们带来了无限的便捷,满足了多元化的消费需求。国内市场上信用卡种类各异,让人眼花缭乱。哪个银行信用卡额度高?如何才能申请到更高额度的卡片呢?今天平安信用卡小编,就来为大家详信用卡为什么有两张卡(办信用卡怎么收到2张)之前,朋友申请了光大银行的阳光信用卡银联金卡,但是,收到卡的时候发现竟然有两张,一张是银联标识的卡,另外一张是带有VISA标识的全币种卡。为什么会出现申请一张信用卡,邮寄到手变两张华夏银行信用卡为什么是两张(可以办两张华夏信用卡吗)一华夏简介华夏银行于1992年10月在北京成立,于2003成为全国第五家上市银行,但这家银行的存在感并不强。办理难度提额难度风控程度羊毛力度二信用卡等级华夏信用卡分为普卡金卡钛金卡如何查信用卡额度(可用额度比信用额度少)虽然没有收到信用卡,但是急迫的想知道自己的信用卡额度怎么办?下面小编就带大家看看广发银行信用卡怎么查额度?一广发银行信用卡一般额度是多少广发银行信用卡额度是根据申请人的资质决定的,pos机多少钱一台(拉卡拉0。38费率安全吗)支付行业每天都有一堆人投诉这投诉那的,侧面反映了行业内的乱象有始无终。拉K拉客服骚扰,短信轰炸用户刘先生在网页中申请拉K拉pos机,客服联系中并没有告知需要服务费。当用户收到货激活pos机是什么(个人办理pos机利弊)一什么是限额?POS机限额常见与对私账户结算的商户,如单笔交易限额,或单卡交易限额,或每日到账限额等根据收单方不同的风险控制政策做出不同的限额要求以降低非法交易的风险。二什么是跳码