40、字符串的組合

題目:輸入一個(gè)字符串(可能有重復(fù)字符),求它字符的所有組合。

例:
輸入abc,結(jié)果為[bc, a, ab, b, ac, c, abc]
輸入abac,結(jié)果為[aa, a, bc, ab, ac, b, aac, c, abc, aab, aabc]

解法一:每個(gè)字符都可選,可不選,即每個(gè)字符有兩種可能,以abc舉例,抽象成一棵二叉樹,如下(帶角標(biāo)的代表不選,如 a'代表不選a ):


image.png(a'代表不選a)
  private List<String> combination(String str) {
        List<String> result = new ArrayList<>();
        if (str == null || str.length()==0) return result;
        dfs(result, "", 0, str.toCharArray());
        HashSet<String> set = new HashSet<>(result);    // 去重
        return new ArrayList<>(set);
    }

    /**
     * result 結(jié)果集
     * sb 當(dāng)前組合起來(lái)的字符串
     * index 當(dāng)前正在操作的字符的下標(biāo)
     * str 字符數(shù)組
     **/
    private void dfs(List<String> result, String sb, int index, char[] str) {
        // 當(dāng)最后一個(gè)字符已經(jīng)操作完后,將組合后的字符串放入結(jié)果集
        if (index >= str.length) {
            if (sb.length() > 0) {      // 去除所有字符都不選的結(jié)果,即空字符串""
                // 將結(jié)果重新排序,方便后面去重
                char[] chars = sb.toCharArray();
                Arrays.sort(chars);
                result.add(new String(chars));
            }
            return;
        }
        // 不選index所在的字符
        dfs(result, sb, index + 1, str);
        // 選index所在的字符
        dfs(result, sb + str[index], index + 1, str);
    }

解法二:利用二進(jìn)制,每個(gè)字符代表二進(jìn)制中的一位,0代表不選,1代表選。例如:abc,用111表示,即用7=2^3-1 表示;a=100,b=010,c=001,ab=110,以此類推。共有7=2^3-1種可能。

private List<String> combination(String str) {
        List<String> result = new ArrayList<>();
        if (str == null || str.length()==0) return result;
        int max = (int) (Math.pow(2, str.length()) - 1);
        char[] chars = str.toCharArray();
        // i從1開始,去掉空字符串的可能
        for (int i = 1; i <= max; ++i) {
            getString(i, result, chars);
        }
        HashSet<String> strings = new HashSet<>(result);
        return new ArrayList<>(strings);
    }

    // 根據(jù)num確定哪些字符被選了,例如以abc舉例,3=011,選了b和c
    private void getString(int num, List<String> result, char[] str) {
        // index表示當(dāng)前在處理的字符下標(biāo),從最后一位字符開始
        int index = str.length - 1;    
        // 保存選中的字符下標(biāo),方便后面組合字符串
        List<Integer> indexes = new ArrayList<>();  
        // 將num模2,如果結(jié)果為1,說(shuō)明index所在的字符被選中,將index放入indexes。
        // 然后將num右移一位(就是除2啦),同時(shí)將index-1指向前一個(gè)字符。
        // 重復(fù)此過(guò)程,直到num==0
        while (num != 0) {
            if (num % 2 == 1) {
                indexes.add(index);
            }
            num /= 2;
            --index;
        }
        StringBuilder sb = new StringBuilder();
        // 將選中的字符組合成字符串,并對(duì)字符重新排序,方便去重,然后放入結(jié)果集中
        for (int i = indexes.size() - 1; i >= 0; --i) {
            sb.append(str[indexes.get(i)]);
        }
        char[] chars = sb.toString().toCharArray();
        Arrays.sort(chars);
        result.add(new String(chars));
    }
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • 專業(yè)考題類型管理運(yùn)行工作負(fù)責(zé)人一般作業(yè)考題內(nèi)容選項(xiàng)A選項(xiàng)B選項(xiàng)C選項(xiàng)D選項(xiàng)E選項(xiàng)F正確答案 變電單選GYSZ本規(guī)程...
    小白兔去釣魚閱讀 10,701評(píng)論 0 13
  • 在C語(yǔ)言中,五種基本數(shù)據(jù)類型存儲(chǔ)空間長(zhǎng)度的排列順序是: A)char B)char=int<=float C)ch...
    夏天再來(lái)閱讀 4,090評(píng)論 0 2
  • 基礎(chǔ)命令 主要的命令和快捷鍵 Linux系統(tǒng)命令由三部分組成:cmd + [options]+[operation...
    485b1aca799e閱讀 1,234評(píng)論 0 0
  • 排列問題 問題描述:已知字符串的中字符是互不相同的,求所有可能的字符排列。例如輸入序列ab,則得到的所有可能序列應(yīng)...
    進(jìn)擊的Lancelot閱讀 485評(píng)論 0 6
  • 字符串的全組合 題目描述: 輸入一個(gè)字符串,輸出該字符串中字符的所有組合。舉個(gè)例子,如果輸入abc,它的組合有a、...
    MinoyJet閱讀 3,542評(píng)論 0 1

友情鏈接更多精彩內(nèi)容