++递归 字符串全排列_一文看懂全排列算法
作者 | Cooper Song
责编 | 伍杏玲
所谓全排列,就是把一堆字符按照一定的顺序排列起来,所有可能的组合。
举个简单的例子,"123"的全排列为"123"、"132"、"213"、"231"、"312"、"321"。
使用库函数进行全排列
C++的头文件中实现了全排列,即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
#include
using namespace std;
string str;
int main
{
getline(cin,str);
//先进行排序使之字典序最小
sort(str.begin,str.end);
cout<
do
{
cout<
}while(next_permutation(str.begin,str.end));
return 0;
}
手撕全排列
可是如何用编程实现这一过程呢?本文就教大家使用深搜回溯算法实现全排列。代码如下:
#include #include #include using namespace std;string str;string temp;vector vis;void dfs(int cnt,int n){if(cnt==n){cout<return;}for(int i=0;i{if(vis[i]==false){temp.push_back(str[i]);vis[i]=true;dfs(cnt+1,n);vis[i]=false;temp.pop_back;}}}int main{getline(cin,str);sort(str.begin,str.end);int n=str.size;vis.resize(n);dfs(0,n);return 0;}
我们把产生的排练暂存在字符串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。
++递归 字符串全排列_一文看懂全排列算法相关推荐
- angular 字符串转换成数字_一文看懂Python列表、元组和字符串操作
好文推荐,转自CSDN,原作星辰StarDust,感觉写的比自己清晰-大江狗荐语. 序列 序列是具有索引和切片能力的集合. 列表.元组和字符串具有通过索引访问某个具体的值,或通过切片返回一段切片的能力 ...
- 怎么看电脑系统是win几_一文看懂arm架构和x86架构有什么区别
一文看懂arm架构和x86架构有什么区别 本文主要介绍的是arm架构和x86架构的区别,首先介绍了ARM架构图,其次介绍了x86架构图,最后从性能.扩展能力.操作系统的兼容性.软件开发的方便性及可使用 ...
- 用户画像标签维度_一文看懂用户画像标签体系(包括维度、应用场景)
一文看懂用户画像标签体系(包括维度.应用场景) 互联网相关企业在建立用户画像时一般除了基于用户维度(userid)建立一套用户标签体系外,还会基于用户使用设备维度(cookieid)建立相应的标签体系 ...
- 判别两棵树是否相等 设计算法_一文看懂生成对抗网络 - GANs?(附:10种典型算法+13种应用)...
生成对抗网络 – GANs 是最近2年很热门的一种无监督算法,他能生成出非常逼真的照片,图像甚至视频.我们手机里的照片处理软件中就会使用到它. 本文将详细介绍生成对抗网络 – GANs 的设计初衷.基 ...
- 无处 不在的无线智能——6g 的关键驱动与研究挑战_一文看懂什么是 6G
原标题:一文看懂什么是 6G 2020年行将结束,随着5G网络的建设推进,以及3GPP R16版本的冻结,越来越多的人将关注焦点转移到6G身上. 7月14日,韩国三星电子发布了白皮书<下一代超连 ...
- mysql删除分表键_一文看懂 MySQL 分区和分表,提高表增删改查效率
原标题:一文看懂 MySQL 分区和分表,提高表增删改查效率 作者:冯帅,精通Oracle. MySQL. 擅长异构数据库数据同步及迁移.数据库的设计和调优,对高可用方案有深入研究. MySQL分区和 ...
- 天线巴伦制作和原理_一文看懂巴伦(功能原理、性能参数、基本类型)
原标题:一文看懂巴伦(功能原理.性能参数.基本类型) 巴伦(英语为balun)为一种三端口器件,或者说是一种通过将匹配输入转换为差分输出而实现平衡传输线电路与不平衡传输线电路之间的连接的宽带射频传输线 ...
- java rest 序列化_一文看懂Java序列化
一文看懂Java序列化 简介 首先我们看一下wiki上面对于序列化的解释. 序列化(serialization)在计算机科学的数据处理中,是指将数据结构或对象状态转换成可取用格式(例如存成文件,存于缓 ...
- 小米iot业务_一文看懂小米2019上半年财报:IoT平台连接设备达1.96亿台
8月20日,小米发布了今年第二季度财报.第二季度营收519.51亿元,同比增长14.8%;调整后净利润为36.4亿元,同比增长71.7%. 除了盈利能力的增强,这个季度小米在智能手机业务上的调整初见成 ...
最新文章
- html背景音乐demo,music.html
- java 滚轮页面缩放_急..JAVA 在画布上画拖动滚动条可扩大缩小的长方形
- JavaScript比较中应使用哪个等于运算符(== vs ===)?
- 人脸检测 RetinaFace
- 腾讯云linux服务器怎么使用图形化界面_自己搭建一个自动签到和远程下载的服务器...
- Sqlyog的安装使用
- 上海立信会计学院_上海立信会计学院分数线_上海立信_立信会计_立信会计学院...
- 【渝粤题库】陕西师范大学400011 思想政治教育学科教学论 作业(专升本)
- 一天一道算法题--6.15--卡特兰数
- java有什么压力_编程语言的心智负担!你学编程得有多大的压力快来测试一下...
- ZooKeeper 3.4.5 分布式环境搭建
- 计算机网络实验二抓包协议分析,计算机网络实验-使用Wireshark分析TCP和UDP协议...
- oracle使用索引和不使用索引性能分析
- 霍夫曼编码实验matlab,哈夫曼编码 MATLAB程序
- 大一学科不挂科 速冲方法推荐(高数上+高数下+线代+大物)
- 大数据时代中国“破障”
- webbrowser点击网页内部链接阻止从IE打开
- 迁移服务器和切换域名
- SEO优化方案怎么做(seo推广优化方案)
- 【一元多项式算法】设一个一元多项式采用带头结点的单链表存储,所有结点 按照升幂方式链接。设计一个算法,求两个多项式 A 和 B 的乘积,结果多项式 C 存放在新辟的空间中。
热门文章
- Linux 下 Git 的源码安装
- RPC(远程过程调用)原理及应用
- matlab gui怎样将结果保存_Matlab将工作区变量保存到文件中的方法
- python3 单例模式_当python,单例模式,多例模式,一次初始化遇到一起
- angular input_可视化的 Angular 响应式编程
- 学python后到底能干什么-普通小白学会Python到底具体能做什么呢?
- python全套完整教程-Python语言基础50课 全套完整版(含doc版)
- python是干什么用的-使用Python究竟可以做什么?下面是Python的3个主要应用
- python100例详解-Python类和实例详解
- 自学python好找工作么-转行去培训班学Python好找工作吗?老男孩教育