java數(shù)組排列組合問題匯總
面試或筆試中,多次遇到以下4個(gè)關(guān)于排列組合的手撕算法,這里做個(gè)筆記,方法日后查閱:
1. 無重復(fù)元素的數(shù)組,求全排列;
2. 有重復(fù)元素的數(shù)組,求全排列;
3. 無重復(fù)元素的數(shù)組,求組合【子集】;
4. 有重復(fù)元素的數(shù)組,求組合;
以上四類題,可以用統(tǒng)一的模板實(shí)現(xiàn),如下所示:
/*
*【組合&&排列】
*把一個(gè)數(shù)組里的數(shù)組合全部列出,比如1和2列出來為1,2,12,21.
*這個(gè)題目可以擴(kuò)展成四個(gè):
*1.無重復(fù)數(shù)字的數(shù)組,求組合
*2.有重復(fù)數(shù)字的數(shù)組,求組合
*3.無重復(fù)數(shù)字的數(shù)組,求全排列
*4.有重復(fù)數(shù)字的數(shù)組,求全排列
*【通用思路(遞歸)】:
*定義一個(gè)函數(shù):從候選集candicate中選取一個(gè)組合prefix
*每次從候選集candicate中remove一個(gè)數(shù),并加入前綴prefix,打印prefix;
*再遞歸從新的candicate中remove下一個(gè)數(shù),并加入prefix
*【對于重復(fù)的控制】
*采用hashset保存prefix,打印之前,判斷hashset中是否包含當(dāng)前生成的prefix,
*沒有則打印,并加入hashset;有則不打印
*【組合--》排列】
*只需在打印前加一個(gè)判斷:若候選集candicate為空,表示遍歷完一次,生成一個(gè)排列,可打印
*/
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減少一個(gè)數(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-10
Java中LinkedList詳解和使用示例_動(dòng)力節(jié)點(diǎn)Java學(xué)院整理
LinkedList 是一個(gè)繼承于AbstractSequentialList的雙向鏈表。它也可以被當(dāng)作堆棧、隊(duì)列或雙端隊(duì)列進(jìn)行操作。接下來通過示例代碼給大家詳細(xì)介紹java中l(wèi)inkedlist的使用,需要的朋友參考下吧2017-05-05
使用Jenkins一鍵打包部署SpringBoot項(xiàng)目的步驟詳解
任何簡單操作的背后,都有一套相當(dāng)復(fù)雜的機(jī)制,本文將以SpringBoot應(yīng)用的在Docker環(huán)境下的打包部署為例,詳細(xì)講解如何使用Jenkins一鍵打包部署SpringBoot應(yīng)用,文中通過圖文結(jié)合講解的非常詳細(xì),需要的朋友可以參考下2023-11-11
詳解java CountDownLatch和CyclicBarrier在內(nèi)部實(shí)現(xiàn)和場景上的區(qū)別
這篇文章主要介紹了詳解java CountDownLatch和CyclicBarrier在內(nèi)部實(shí)現(xiàn)和場景上的區(qū)別,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-05-05
Maven及Springboot配置JDK版本,編碼,源碼打包等方式
這篇文章主要介紹了Maven及Springboot配置JDK版本,編碼,源碼打包等方式,具有很好的參考價(jià)值,希望對大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-12-12
LinkedList學(xué)習(xí)示例模擬堆棧與隊(duì)列數(shù)據(jù)結(jié)構(gòu)
這篇文章主要介紹了LinkedList學(xué)習(xí)示例,模擬一個(gè)堆棧與隊(duì)列數(shù)據(jù)結(jié)構(gòu),大家參考使用吧2014-01-01

