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

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

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

text是什么意思(text的动词意思)上月,被誉为短信之父的芬兰工程师马蒂马克能因病去世。31年前,正是由他设计和提出了现代短信(ShortMessageService,简称SMS)系统的雏形。之后,这些被传递的字符不点缀是什么意思(绣缀是什么意思)一年级一看拼音,写字词。xionishunishugupnggu苹二选择正确的序号填在横线上。wnyudngxnhixngxingxiochun二现代飒什么意思(又酷又飒什么意思)你会解释吗?今年十大流行语后浪飒内卷打工人一人民至上,生命至上为了保护人民生命安全和身体健康可以不惜一切代价。二逆行者指在2020年新冠疫情阻击战中涌现出的先进个人和集体。广大医务鱼肚是什么(鱼肚是什么样的图片)花胶花胶,又称鱼胶,是鱼膘的干品,即鱼在水中调节升降的器官,并非指鱼的肚子或者胃。四大极品鲍参翅肚中的肚即为鱼胶。我国食用花胶有悠久的历史。1600多年前,在北魏贾思勰齐民要术中就雅典娜是什么神(阿尔忒弥斯是什么神)雅典娜是宙斯所生,阿瑞斯的姐姐。文武双全的智慧女战神与莽撞无脑的阿瑞斯形成鲜明的对比。在雅典帕特农神庙纪念雅典娜。英姿飒爽的一身戎装的武女战士,头戴战盔,身着胸甲,左手持盾,右手执鹤立鸡群是什么意思(有鹤立鸡群这个词吗)鹤立鸡群在我国南北朝时期的魏晋时期,有一个人叫吉韶。他是魏晋之际竹林七贤之一吉康的儿子,他长的体态魁梧,相貌英俊,聪明伶俐,在他的同龄人中显得非常地出众。在晋慧帝当政时,吉韶在朝廷四面楚歌是什么意思(四面楚歌打一种动物)肯尼亚内罗毕国家公园,一头落单的雄狮遭到一群水牛的团团围攻,凶猛的捕食者差点沦为了孤立无援的猎物。11月3日,摄影师LamakSheikh在内罗毕国家公园驱车旅游,撞见了这头运气不MercedesBenz深耕台湾开启极智未来2015年台湾宾士创下自台湾分公司成立以来,史上最高新车挂牌数,以近21,500辆挂牌数高达11大幅成长,创下全新里程碑,也超越整体车市表现。感谢全台车主支持,2016年将秉持深耕天梭哪里的(中国名表官方)又是一年七夕!多少人在这个节日挠破了脑袋,只为给心爱的ta送上一份称心意的礼物?选对礼物可以让你事半功倍!如果你在犹豫送暗恋女神女友妻子什么礼物,手表往往是最万能的选择。手表谐音表什么是电子邮件地址(电子邮件地址怎么弄)电子邮箱在我们的日常工作中使用的频率很高,用于收发邮件传输文件。电子邮箱地址在格式上是有规律的,其中包含一个特殊字符。那么,这个究竟代表什么呢?每一个电子邮箱都有独立的地址进行标识energy是什么意思(gate是什么意思)今天让我们一起谈谈能量,我们都在说能量,但能量到底是什么?它和普通人的生活有什么关系?火是能量,水也是能量,风也是能量,太阳光也是能量,电是能量,核能也是能量,就连双手互相摩擦也可
二手房过户费怎么算(保定二手房过户费怎么算)现在,郑州新房价格是芝麻开花节节高。而二手房价比新房低很多,买了基本可以拎包入住。那么二手房过户交易需要缴纳那些费用呢?接下来,超彩房掌柜就来说说二手房过户费是多少?希望能够帮组到蜂王浆的吃法(蜂王浆怎么吃效果好?)蜂王浆的吃法(蜂王浆怎么吃效果好?)蜂王浆,有很多食用方法,但是哪种食用方法是最好的呢?下面,田野牧蜂带您一起来了解下吃蜂王浆的各种方法,并介绍哪种吃法最好。1。直接食用直接服用蜂上海到黄山多少公里(上海到歙县多少公里)有句话说念念不忘,必有回想。可以说对于徽州来说,早在很久之前就游过了不好地方,现在的每次都到基本都是在填补之前记忆的空缺,我不知道别人的记忆会怎样,对于我来说,第一次是新鲜的,但是超级亚洲(亚洲一共有多少个国家)超级亚洲(亚洲一共有多少个国家)环球网报道记者翟潞曼7月5日是中国网络名人环球行韩国站活动代表团抵达首尔进行访问的第二日,来自中国的大V们参观了三星DLight宣传馆SMTOWN及亚洲多少个国家(亚洲一共有多少个国家?)亚洲竟然仅有4个发达国家,为什么没有中国?中国能成为第5个吗?世界上总共有233个国家,发达国家大部分位于欧洲以及北美洲。亚洲仅有四个发达国家,分别是日本新加坡韩国以及以色列。为什全球共有多少个国家(当今世界公认5大强国)第42。马来西亚(亚洲第14)马来西亚地处热带东南亚,又被称为大马,是亚洲四小虎之一,全国由马来半岛和婆罗洲北部区两个部分组成,与印尼新加坡等国隔海相望,小国文莱则嵌入马来西亚领土台式电脑功率(台式电脑电源功率多大才合适?)台式电脑功率(台式电脑电源功率多大才合适?)组装一台电脑,我们需要先挑选好硬件,搭配硬件最关键点就是CPU和主板兼容性,硬件之间的均衡性电源功率等,均需要考虑周到。那么台式电脑电源美容院脱毛多少钱(医院脱毛一般需要多少钱)今天说到脱毛费用这个话题应该也是大家比较的话题之一了,怎么说呢,看大家怎么选择吧,一般如果大家选择去正规的医院或专业的美容机构的话,那费用就比较高昂了,为什么这么说呢,因为一般像这脱毛需要多少钱(美容院脱毛一般多少价格)作为一个从小毛发就比同龄人多的女汉纸,给你们分享一下脱毛的问题!其实身体每个部位的皮肤都不一样,所以都要非常仔细的去处理。。先来说一说我的血泪史吧腋下皮肤比较薄,之前用刮刀经常刮破做脱毛多少钱(激光脱毛要做几次多少钱)光滑有弹性的皮肤是每个人都想拥有的,拥有这样的皮肤可以让人变得更加美丽自信,特别是生活中,有很多人脸上的手发很重这也是最让人困扰的地方,所以会有一些爱美的女生通过脸部脱毛手术摆脱这7位数qq多少钱(7位qq号专卖店)文小凡如果提到我们最为熟悉也是最早的社交软件的话,那么大家第一时间想到的肯定是出自腾讯公司的QQ。QQ可以说是承载了一代人的青春,是一个时代的里程碑,并且还将这个世界都缩小到了小小