回溯算法之-组合总和

达叔

回溯算法模版

首先上一套回溯算法模版,很多回溯算法都可以使用该模版解决

  public List<List<Integer>> problem(参数不定) {
        List<List<Integer>> res = new ArrayList<>();
        help(参数不定);
        return res;
    }

    public void help(参数不定){
        if(满足条件){
          加入解集中
        return
        }
        for(可以选择的数据集合){
            // 做选择
            list.add(i);
            help();
            // 回溯
            list.remove(list.size()-1);
        }
    }

leetcode 39 组合总和

给定一个无重复元素的数组 candidates 和一个目标数 target ,找出 candidates 中所有可以使数字和为 target 的组合。
candidates 中的数字可以无限制重复被选取。解集不能包含重复的组合

两个要点已经被黑体标注,数字可重复选取比较容易理解,即假设第一次选择完2之后,还可以继续选择2;对于不包含重复组合规定,我们先来看下怎么样的组合算重复,假设target = 5,candidates为2,3,那么【2,3】为一组,【3,2】同样为一组,但【3,2】算作与【2,3】重复,因为组合问题不关心解集中元素顺序,即如果两个解集中元素相同,但顺序不同,那么这两个解就算重复。

接下来看下代码,套模版

public List<List<Integer>> combinationSum(int[] candidates, int target) {
        List<List<Integer>> res = new ArrayList<>();
      Arrays.sort(candidates); // 这里的排序非必须
        help(res, candidates, target, new ArrayList(), 0);
        return res;
    }
    private void help(List<List<Integer>> res, int[] candidates, int target, ArrayList arrayList, int index){
       // 对于不满足条件的直接return,俗称剪枝
        if(target < 0) {
            return;
        }
      // 满足条件
        if(target == 0) {
            res.add(new ArrayList(arrayList));
            rerurn;
        }
        // 从待选择列表中选择,注意这里index的作用在于不能让你再选择之前已经选过的集合元素了
// 比较难理解,举个例子,假设candidates为 2,3,4
// index=0,选择2,那么剩余可选为3,4
// index = 1,选择3,剩余可选为4
// index = 2,选择4,剩余可选为空
// 为什么要只让他选择index后面的元素呢?这是为了排重,假设2,3,4是一个解集,当选择4时,如果让他可以随意选取,那么就会选择到4,2,3;4,3,2的情况,所以这里限制只能选择上一轮没有选择的元素
        for(int i=index;i<candidates.length;i++){
            int num = candidates[I];
            arrayList.add(num);
          // help函数最后一个参数传i,代表着下一次循环还可以继续选择该元素
            help(res, candidates, target - num, arrayList, i);
            arrayList.remove(arrayList.size()-1);
        }
    }

代码还有可以优化的一些地方,比如剪枝代码可以拿到for循环中提前判断,存储解集的集合可以使用双端队列减少代码复杂度

leetcode 40 组合总和2

总体描述与上一题相同,只有约束条件有改变,**candidates 中的每个数字在每个组合中只能使用一次,并且candidates可能包含重复数字

同样,套模版

public List<List<Integer>> combinationSum2(int[] candidates, int target) {
        List<List<Integer>> res = new ArrayList<>();
        // 这里的排序是必须的
        Arrays.sort(candidates);
        help(res, candidates, target, new ArrayList(), 0);
        return res;
    }

    public void help(List<List<Integer>> res, int[] candidates, int target, List<Integer> list, int index){
        if(target <0){
            return;
        }
        if(target == 0) {
            res.add(new ArrayList<>(list));
            return;
        }
        for(int i =index;i<candidates.length;i++){
      
            if(i>index && candidates[i]==candidates[i-1]){
                continue;
            }
            int num = candidates[I];
            list.add(num);
            // 每个数字只能使用一次,所以是i+1
            help(res,candidates,target - num, list, i+1);
            list.remove(list.size()-1);
        }
    } 

leetcode 216 组合总和3

找出所有相加之和为 n 的 k 个数的组合。组合中只允许含有 1 - 9 的正整数,并且每种组合中不存在重复的数字。
解集不能包含重复的组合

和组合2问题基本相同,唯一的区别在于加了限制解集只能包含k个数

直接套模版

public List<List<Integer>> combinationSum3(int k, int n) {
        List<List<Integer>> res = new ArrayList<>();
        help(res, new ArrayList(), k, n, 1);
        return res;
    }

    public void help(List<List<Integer>> res, List<Integer> list, int k, int n, int index){
        // 这里满足解集需要两个条件
        if(n==0 && list.size()==k) {
            res.add(new ArrayList(list));
            return;
        }
        // 剪枝的情形也多了一个
        if(n <0 || list.size() >k){
            return;
        }
        for(int i=index; i<10; i++){
            list.add(i);
            help(res, list, k, n-i, i+1);
            list.remove(list.size()-1);
        }
    }
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,324评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,303评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,192评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,555评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,569评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,566评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,927评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,583评论 0 257
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,827评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,590评论 2 320
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,669评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,365评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,941评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,928评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,159评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,880评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,399评论 2 342

推荐阅读更多精彩内容