java數(shù)組排列組合問題匯總
面試或筆試中,多次遇到以下4個關(guān)于排列組合的手撕算法,這里做個筆記,方法日后查閱:
1. 無重復(fù)元素的數(shù)組,求全排列;
2. 有重復(fù)元素的數(shù)組,求全排列;
3. 無重復(fù)元素的數(shù)組,求組合【子集】;
4. 有重復(fù)元素的數(shù)組,求組合;
以上四類題,可以用統(tǒng)一的模板實現(xiàn),如下所示:
/* *【組合&&排列】 *把一個數(shù)組里的數(shù)組合全部列出,比如1和2列出來為1,2,12,21. *這個題目可以擴展成四個: *1.無重復(fù)數(shù)字的數(shù)組,求組合 *2.有重復(fù)數(shù)字的數(shù)組,求組合 *3.無重復(fù)數(shù)字的數(shù)組,求全排列 *4.有重復(fù)數(shù)字的數(shù)組,求全排列 *【通用思路(遞歸)】: *定義一個函數(shù):從候選集candicate中選取一個組合prefix *每次從候選集candicate中remove一個數(shù),并加入前綴prefix,打印prefix; *再遞歸從新的candicate中remove下一個數(shù),并加入prefix *【對于重復(fù)的控制】 *采用hashset保存prefix,打印之前,判斷hashset中是否包含當(dāng)前生成的prefix, *沒有則打印,并加入hashset;有則不打印 *【組合--》排列】 *只需在打印前加一個判斷:若候選集candicate為空,表示遍歷完一次,生成一個排列,可打印 */ package xh.offer.practice; import java.util.Arrays; import java.util.HashSet; import java.util.LinkedList; import java.util.List; public class listAllGroup{ public static void main(String[] args) { String [] array = {"1","2"}; String [] repeate = {"1","2","1"}; listAllGroup test = new listAllGroup(); System.out.println("**********no repeate list*******"); test.listAllNoRepeate(Arrays.asList(array),"");//初始化prefix = "" System.out.println("**********repeate list*******"); HashSet<String> noRepeateSet = new HashSet<String>(); test.listAllRepeate(Arrays.asList(repeate), "", noRepeateSet); System.out.println("**************no repeate premutation********************"); test.premutationNoRepeate(Arrays.asList(array),""); System.out.println("*********************repeate premutation**********************"); HashSet<String> repeateSet = new HashSet<String>(); test.premutationRepeate(Arrays.asList(repeate),"", repeateSet); } //無重復(fù)的組合 public void listAllNoRepeate(List<String> candicate,String prefix){ if(prefix.length() != 0) System.out.println(prefix);//結(jié)果長度不為0,則打印輸出該組合 for(int i = 0;i < candicate.size();i++){ //將list轉(zhuǎn)換成linklist鏈表,方便操作 List<String> tempList = new LinkedList<String>(candicate); //templist減少一個數(shù)字,并暫存templist中去除的數(shù)字 String tempString = (String) tempList.remove(i); //遞歸 listAllNoRepeate(tempList,prefix + tempString); } } //有重復(fù)的組合,加入hashset public void listAllRepeate(List<String> candicate,String prefix,HashSet<String> res){ if(prefix.length() != 0 && !res.contains(prefix)){ System.out.println(prefix); res.add(prefix); } for(int i = 0;i < candicate.size();i++){ List<String> tempList = new LinkedList<String>(candicate); String tempString = tempList.remove(i); listAllRepeate(tempList, prefix+tempString, res);//遞歸 } } //無重復(fù)的全排列,加入判斷candicate.size() == 0 public void premutationNoRepeate(List<String> candicate,String prefix){ if(candicate.size() == 0){ System.out.println(prefix); } for(int i = 0;i < candicate.size();i++){ List<String> tempList = new LinkedList<String>(candicate); String tempString = tempList.remove(i); premutationNoRepeate(tempList,prefix+tempString); } } //有重復(fù)的全排列,加入hashset輔助判斷輸出 public void premutationRepeate(List<String> candicate,String prefix,HashSet<String> res) { if(candicate.size() == 0 && !res.contains(prefix)){ System.out.println(prefix); res.add(prefix); } for(int i = 0;i < candicate.size();i++){ List<String> tempList = new LinkedList<String>(candicate); String tempString = tempList.remove(i); premutationRepeate(tempList, prefix+tempString, res); } } }
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
在Mac OS上安裝Java以及配置環(huán)境變量的基本方法
這篇文章主要介紹了在Mac OS上安裝Java以及配置環(huán)境變量的基本方法,包括查看所安裝Java版本的方法,需要的朋友可以參考下2015-10-10Java中LinkedList詳解和使用示例_動力節(jié)點Java學(xué)院整理
LinkedList 是一個繼承于AbstractSequentialList的雙向鏈表。它也可以被當(dāng)作堆棧、隊列或雙端隊列進行操作。接下來通過示例代碼給大家詳細(xì)介紹java中l(wèi)inkedlist的使用,需要的朋友參考下吧2017-05-05使用Jenkins一鍵打包部署SpringBoot項目的步驟詳解
任何簡單操作的背后,都有一套相當(dāng)復(fù)雜的機制,本文將以SpringBoot應(yīng)用的在Docker環(huán)境下的打包部署為例,詳細(xì)講解如何使用Jenkins一鍵打包部署SpringBoot應(yīng)用,文中通過圖文結(jié)合講解的非常詳細(xì),需要的朋友可以參考下2023-11-11詳解java CountDownLatch和CyclicBarrier在內(nèi)部實現(xiàn)和場景上的區(qū)別
這篇文章主要介紹了詳解java CountDownLatch和CyclicBarrier在內(nèi)部實現(xiàn)和場景上的區(qū)別,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-05-05Maven及Springboot配置JDK版本,編碼,源碼打包等方式
這篇文章主要介紹了Maven及Springboot配置JDK版本,編碼,源碼打包等方式,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2023-12-12LinkedList學(xué)習(xí)示例模擬堆棧與隊列數(shù)據(jù)結(jié)構(gòu)
這篇文章主要介紹了LinkedList學(xué)習(xí)示例,模擬一個堆棧與隊列數(shù)據(jù)結(jié)構(gòu),大家參考使用吧2014-01-01