tft每日頭條

 > 生活

 > 全排列算法描述

全排列算法描述

生活 更新时间:2024-09-04 10:21:11

全排列算法描述(一文看懂全排列算法)1

作者 | Cooper Song

責編 | 伍杏玲

所謂全排列,就是把一堆字符按照一定的順序排列起來,所有可能的組合。

舉個簡單的例子,"123"的全排列為"123"、"132"、"213"、"231"、"312"、"321"。

全排列算法描述(一文看懂全排列算法)2

使用庫函數進行全排列

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>

using namespace std;

string str;

int main

{

getline(cin,str);

//先進行排序使之字典序最小

sort(str.begin,str.end);

cout<<"其全排列為:"<<endl;

do

{

cout<<str<<endl;

}while(next_permutation(str.begin,str.end));

return 0;

}

全排列算法描述(一文看懂全排列算法)3

全排列算法描述(一文看懂全排列算法)4

手撕全排列

可是如何用編程實現這一過程呢?本文就教大家使用深搜回溯算法實現全排列。代碼如下:

#include <iostream>#include <vector>#include <algorithm>using namespace std;string str;string temp;vector<bool> vis;void dfs(int cnt,int n){if(cnt==n){cout<<temp<<endl;return;}for(int i=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;}}}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。

,

更多精彩资讯请关注tft每日頭條,我们将持续为您更新最新资讯!

查看全部

相关生活资讯推荐

热门生活资讯推荐

网友关注

Copyright 2023-2024 - www.tftnews.com All Rights Reserved