劍指 Offer 38. 字串的排列
輸入一個字串,列印出該字串中字元的所有排列。
你可以以任意順序返回這個字串陣列,但裡面不能有重複元素。
示例:
輸入:s = "abc"
輸出:["abc","acb","bac","bca","cab","cba"]
限制:
- 1 <= s 的長度 <= 8
回溯法
遞迴思路:
- 如果c[i]在set裡面,則進行剪枝
- 將c[i]固定在第X位,方便進行交換,然後開啟第x+1的下層遞迴,直到最後還原之前的交換
然後關於固定字元,我覺得這位力友(guyue)解釋的比較清楚。
通過交換來固定某個位置的元素這個思路真的太棒了,就 abc 這個字串來說,第一個位置可以放 a 或者 b 或者 c,但是如果確定要放某個字元,比如第一個位置放 a,那麼第二個位置就只能放 b 或者 c;如果第一個位置放 b,那麼第二個位置就只能放 a 或者 c;如果第一個位置放 c,那麼第二個位置就只能放 a 或者 b;當把某個字元移動到第一位以後,暫時第一位的字元就固定住了,這時再去確定第二個位置的元素,並且此時第一個位置的元素不會再出現在後面的位置上,依次類推直到確定所有位置的元素,再往前回溯確定每個位置上其他可能出現的元素。
class Solution {
List<String> res = new LinkedList<>();
char[] c;
public String[] permutation(String s) {
c = s.toCharArray();
dfs(0);
return res.toArray(new String[res.size()]);
}
void dfs(int x) {
if(x == c.length - 1) {
res.add(String.valueOf(c)); // 新增排列方案
return;
}
HashSet<Character> set = new HashSet<>();
for(int i = x; i < c.length; i++) {
if(set.contains(c[i])) continue; // 重複,因此剪枝
set.add(c[i]);
swap(i, x); // 交換,將 c[i] 固定在第 x 位
dfs(x + 1); // 開啟固定第 x + 1 位字元
swap(i, x); // 恢復交換
}
}
void swap(int a, int b) {
char tmp = c[a];
c[a] = c[b];
c[b] = tmp;
}
}
註釋版本如下:
class Solution {
//為了讓遞迴函式新增結果方便,定義到函式之外,這樣無需帶到遞迴函式的引數列表中
List<String> list = new ArrayList<>();
//同;但是其賦值依賴c,定義宣告分開
char[] c;
public String[] permutation(String s) {
c = s.toCharArray();
//從第一層開始遞迴
dfs(0);
//將字串陣列ArrayList轉化為String型別陣列
return list.toArray(new String[list.size()]);
}
private void dfs(int x) {
//當遞迴函式到達第三層,就返回,因為此時第二第三個位置已經發生了交換
if (x == c.length - 1) {
//將字元陣列轉換為字串
list.add(String.valueOf(c));
return;
}
//為了防止同一層遞迴出現重複元素
HashSet<Character> set = new HashSet<>();
//這裡就很巧妙了,第一層可以是a,b,c那麼就有三種情況,這裡i = x,正巧dfs(0),正好i = 0開始
// 當第二層只有兩種情況,dfs(1)i = 1開始
for (int i = x; i < c.length; i++){
//發生剪枝,當包含這個元素的時候,直接跳過
if (set.contains(c[i])){
continue;
}
set.add(c[i]);
//交換元素,這裡很是巧妙,當在第二層dfs(1),x = 1,那麼i = 1或者 2, 不是交換1和1,要就是交換1和2
swap(i,x);
//進入下一層遞迴
dfs(x + 1);
//返回時交換回來,這樣保證到達第1層的時候,一直都是abc。這裡捋順一下,開始一直都是abc,那麼第一位置總共就3個交換
//分別是a與a交換,這個就相當於 x = 0, i = 0;
// a與b交換 x = 0, i = 1;
// a與c交換 x = 0, i = 2;
//就相當於上圖中開始的三條路徑
//第一個元素固定後,每個引出兩條路徑,
// b與b交換 x = 1, i = 1;
// b與c交換 x = 1, i = 2;
//所以,結合上圖,在每條路徑上標註上i的值,就會非常容易好理解了
swap(i,x);
}
}
private void swap(int i, int x) {
char temp = c[i];
c[i] = c[x];
c[x] = temp;
}
}
其實上面都是K神的思路,個人還是想用模板的方法做吧,因為大佬的思路還真想不出來啊。
class Solution {
/**
該題類似於 全排列2,本題使用set來去除重複元素
除了使用set去重外,還可以對陣列進行排序,使用visited陣列進行剪枝!
*/
Set<String> res = new HashSet();
public String[] permutation(String s) {
backtrack(s.toCharArray(),new StringBuilder(), new boolean[s.length()]);
return res.toArray(new String[0]);
}
// 回溯函式
public void backtrack(char[] ch,StringBuilder sb, boolean[] visitid){
// 終止條件
if(sb.length() == ch.length){
res.add(sb.toString());
return;
}
// 選擇列表
for(int i = 0; i < ch.length; i++){
// 剪枝,如果當前位置的元素已經使用過,則跳過進入下一個位置
if(visitid[i]) continue;
// 做出選擇
sb.append(ch[i]);
// 更新標記
visitid[i] = true;
// 進入下層回溯
backtrack(ch,sb,visitid);
// 撤銷選擇
sb.deleteCharAt(sb.length()-1);
visitid[i] = false;
}
}
}
參考連結: