定制加工
一文看懂全排列算法!
2025-01-02 11:18

作者 | Cooper Song

责编 | 伍杏玲

出品 | CSDN(ID:CSDNnews)

所谓全排列,就是把一堆字符按照一定的顺序排列起来,所有可能的组合。

举个简单的例子,"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< bool> 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。

    以上就是本篇文章【一文看懂全排列算法!】的全部内容了,欢迎阅览 ! 文章地址:http://ww.kub2b.com/news/18665.html
     栏目首页      相关文章      动态      同类文章      热门文章      网站地图      返回首页 企库往资讯移动站 http://ww.kub2b.com/mobile/ , 查看更多   
最新文章
华为Mate 60大降价!仅4699元入手华为手机降价「华为Mate 60大降价!仅4699元入手」
促销大降价!现仅需5199元就能入手一款高性能的手机。这款手机的设计风格与以往的Mate系列相似,共有四种颜色可供选择。机身采用
保险资金上调入市比例
  新华社北京4月8日电(记者李延霞、张千千)金融监管总局8日对外发布《关于调整保险资金权益类资产监管比例有关事项的通知》
清明假期北京60个重点商圈客流量同比增长11.4%
北京商报讯(记者 胡静蓉)4月6日,北京市商务局发布清明假期消费市场数据显示,假期三天,市商务局重点监测的百货、超市、专业
我爱我家:去年营收125.4亿元,核心城市市占率继续提高
4月9日,房地产经纪企业我爱我家控股集团有限公司(000560.SZ)发布2024年财报。数据显示,2024年,我爱我家实现住房总交易额(g
华为开发者模式关闭教程:步骤与注意事项华为手机开发者选项在哪里「华为开发者模式关闭教程:步骤与注意事项」
华为手机开发者模式关闭教程很多使用手机的朋友在开发或者测试应用时,需要用到华为手机的开发者模式。但在使用完毕后,出于安全
苹果手机怎么关掉振动苹果手机勿扰模式怎么关闭「苹果手机怎么关掉振动」
苹果手机关闭振动的教程苹果手机提供了多种方式来关闭振动,以满足不同用户的需求。以下是详细的步骤指导:方法一:通过控制中心
窦唯的结局,早已注定
“人潮人海中,有你有我,相遇相识相互琢磨。”窦唯隐退多年,娱乐圈中仍然有他的传说。他是唯一一个在纷乱繁杂的演艺圈里,被用
生化危机4:白金版生化危机4手机版「生化危机4:白金版」
游戏介绍:We've given Resident Evil 4 the Platinum treatment! This update contains the COMPLETE Resident Evil 4 saga
投资者大规模撤出原油看跌基金,创2020年以来最大资金外流
智通财经APP获悉,一只与原油市场下跌挂钩的交易型开放式产品(ETF)刚刚经历了自2020年以来最大规模的资金外流。随着原油价格暴跌
对美加征84%关税!中方坚决反制再出“组合拳”
美国东部时间4月8日,美方将此前宣布的对中国输美产品加征34%所谓“对等关税”,进一步提高50%至84%。国务院关税税则委员会今天