27.字符串的排列 输入一个字符串,按字典序打印出该字符串中字符的所有排列。例如输入字符串abc,则打印出由字符a,b,c所能排列出来的所有字符串abc,acb,bac,bca,cab和cba。 1.递归全排列法: 先从左往右固定一个字符,让后面所有字符依次与其交换位置,然后递归执行,递归返回后继续对刚交换的字符再交换回来,以保持输入字符串不变。以这种方式遍历所有字符。 还有就是注意审题,题目中要求打印要按照字典序进行,针对这个要求,我们可以对得到的全排列进行排序再返回。
import java.util.ArrayList; import java.util.List; import java.util.Collections; public class Solution { public ArrayList<String> Permutation(String str) { List<String> res = new ArrayList<>(); if (str.length() > 9 || str.length()<1) { return (ArrayList)res; } //递归的初始值为(ch数组,空的list,初始下标0) fullPerm(str.toCharArray(),0,res);//调用字符串排列函数 //按字典序打印出该字符串中字符的所有排列 Collections.sort(res);//排序输出.测试才能通过 return (ArrayList)res; } private void fullPerm(char[] ch,int start,List<String> res){ //递归结束的条件就是,第一位和最后一位交换完成 //考虑添加这一组字符串进入结果集中 if(start==ch.length-1){ String val=String.valueOf(ch); //判断一下该集合中是否有重复字符串 if(!res.contains(val)){ res.add(val); return; } } else{ // 第一次循环i与start相等,相当于第一个位置自身交换,关键在于之后的循环, // 之后i !=start,则会交换两个不同位置上的字符,直到start==ch.length-1,进行输出; for(int i=start;i<ch.length;i++){ //保证当输入多个重复字符时,不会重复计算 if(i!=start&&ch[i]==ch[start]){ //如果第一位和非第一位重复了,就不交换了. countinue; } swap(ch,start,i);//每一次,交换首位置和第i个位置的元素 fullPerm(ch,start+1,res);// //递归,第一位固定了,将除了第一位,后面的字符串全排序。 swap(ch,start,i);//交换完成还要换回来,保证第一位不变 } } } } public void swap(char[] ch, int i, int j) { char temp = ch[i]; ch[i] = ch[j]; ch[j] = temp; } }第二个swap用以使得字符数组的顺序回到进入递归前的状态,这样才不会影响外部的遍历顺序。因为在第一次交换后进入递归运算的时候,字符数组的顺序改变了,例如“abc”, start= 0时对应‘a’,j = 1时对应 ‘b’,进行一次交换,此时的字符数组的顺序为 “bac”,从递归返回时,顺序依然是“bac”,则进行第二次交换使得 “bac” -> “abc”,这样在后续才可以进行’a’与’c’的交换,不会落下某一种情况
2.回溯法 也就是利用树去尝试不同的可能性,不断地去字符串数组里面拿一个字符出来拼接字符串,当字符串数组被拿空时,就把结果添加进结果数组里,然后回溯上一层。(通过往数组加回去字符以及拼接的字符串减少一个来回溯。)
