剑指offer-test27

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.回溯法
也就是利用树去尝试不同的可能性,不断地去字符串数组里面拿一个字符出来拼接字符串,当字符串数组被拿空时,就把结果添加进结果数组里,然后回溯上一层。(通过往数组加回去字符以及拼接的字符串减少一个来回溯。)

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值