回溯算法

此为回溯算法刷题的一些代码,希望能给读者以启发;

刷题目录:

46. 全排列

47. 全排列 II

78. 子集

90. 子集 II

39. 组合总和

40. 组合总和 II

216. 组合总和 III

131. 分割回文串

开始前,搬运大佬的解题思路:

解决一个回溯问题,实际上就是一个决策树的遍历过程。你只需要思考 3 个问题:

1、路径:也就是已经做出的选择。

2、选择列表:也就是你当前可以做的选择。

3、结束条件:也就是到达决策树底层,无法再做选择的条件。

如果你不理解这三个词语的解释,没关系,我们后面会用「全排列」和「N 皇后问题」这两个经典的回溯算法问题来帮你理解这些词语是什么意思,现在你先留着印象。

代码方面,回溯算法的框架:

result = []

def backtrack(路径, 选择列表):

    if 满足结束条件:

    result.add(路径)

    return

    for 选择 in 选择列表:

        做选择

        backtrack(路径, 选择列表)

        撤销选择

比如在对数字进行全排列过程中,大家都就会采用回溯法的思想,将其以树的形式表示出来大概就是这样。


从根节点开始,待选[1,2,3]已选[ ]。选择3,则待选为[1,2] 已选[3],往下选择1,待选为[2]已选[1,3],向下只有一个选择,后续不表。然后回退,仅一个已选项,继续回退至待选为[1,2] 已选[3],1已经选过则选2...2支路选择结束回退至根节点重新选择,循环往复。


46. 全排列

给定一个 没有重复 数字的序列,返回其所有可能的全排列。

示例:

输入:[1,2,3]

输出:

[

[1,2,3],

[1,3,2],

[2,1,3],

[2,3,1],

[3,1,2],

[3,2,1]

]

算法一:

public class Solution {

    public List<List<Integer>> permute(int[] nums) {

        int len = nums.length;

        List<List<Integer>> res = new ArrayList<>();

        if (len == 0) {

            return res;

        }

        // 使用哈希表检测一个数字是否使用过

        //Set<Integer> used = new HashSet<>(len);

        Deque<Integer> path = new ArrayDeque<>(len);

        dfs(nums, len, 0, path, res);

        return res;

    }

算法二:(大同小异)

    private void dfs(int[] nums, int len, int depth,

                     Deque<Integer> path,

                     List<List<Integer>> res) {

        if (depth == len) {

            res.add(new ArrayList<>(path));

            return;

        }

        for (int i = 0; i < len; i++) {

            if (!path.contains(nums[i])) {

                //used.add(i);

                path.addLast(nums[i]);

                dfs(nums, len, depth + 1, path, res);

                path.removeLast();

                //used.remove(i);

            }

        }

    }

}



47. 全排列 II

难度中等

给定一个可包含重复数字的序列,返回所有不重复的全排列。

示例:

输入:[1,1,2]

输出:

[

[1,1,2],

[1,2,1],

[2,1,1]

]

class Solution {

    List<List<Integer>> res = new LinkedList<>();

    public List<List<Integer>> permuteUnique(int[] nums) {

        LinkedList<Integer> track=new LinkedList<Integer>();

        int[] used=new int[nums.length];

        backtrack(nums,track,used);

        return res;

    }

    public void backtrack(int[] nums,LinkedList<Integer> track,int[] used)

    {

        if(nums.length==track.size())

        {

            if(!res.contains(track)) res.add(new LinkedList(track));

            return ;

        }

        for(int i=0;i<nums.length;i++)

        {

            if(used[i]==1)

            {

                continue;

            }

            track.add(nums[i]);

            used[i]=1;

            backtrack(nums, track,used);

        // 取消选择

            track.removeLast();

            used[i]=0;

        }

    }

}



78. 子集

难度中等

给定一组不含重复元素的整数数组 nums,返回该数组所有可能的子集(幂集)。

说明:解集不能包含重复的子集。

示例:

输入:nums = [1,2,3]

输出:

[

[3],

[1],

[2],

[1,2,3],

[1,3],

[2,3],

[1,2],

[]

]

class Solution {

    public List<List<Integer>> subsets(int[] nums) {

        List<List<Integer>> res = new ArrayList<>();

        backtrack(0, nums, res, new ArrayList<Integer>());

        return res;

    }

    private void backtrack(int i, int[] nums, List<List<Integer>> res, ArrayList<Integer> tmp) {

        res.add(new ArrayList<>(tmp));

        for (int j = i; j < nums.length; j++) {

            tmp.add(nums[j]);

            backtrack(j + 1, nums, res, tmp);

            tmp.remove(tmp.size() - 1);

        }

    }

}



90. 子集 II

难度中等

给定一个可能包含重复元素的整数数组 nums,返回该数组所有可能的子集(幂集)。

说明:解集不能包含重复的子集。

示例:

输入:[1,2,2]

输出:

[

[2],

[1],

[1,2,2],

[2,2],

[1,2],

[]

]

class Solution {

    public List<List<Integer>> subsetsWithDup(int[] nums) {

        List<List<Integer>> res = new ArrayList<>();

        Arrays.sort(nums);

        backtrack(0, nums, res, new ArrayList<Integer>());

        return res;

    }

    private void backtrack(int i, int[] nums, List<List<Integer>> res, ArrayList<Integer> tmp) {

         res.add(new ArrayList<>(tmp));

        for (int j = i; j < nums.length; j++) {

            if(j>i&&nums[j]==nums[j-1]) continue;

            tmp.add(nums[j]);

            backtrack(j + 1, nums, res, tmp);

            tmp.remove(tmp.size() - 1);

        }

    }

}



39. 组合总和

难度中等

给定一个无重复元素的数组 candidates 和一个目标数 target ,找出 candidates 中所有可以使数字和为 target 的组合。

candidates 中的数字可以无限制重复被选取。

说明:

所有数字(包括 target)都是正整数。

解集不能包含重复的组合。 

示例 1:

输入:candidates =[2,3,6,7],target =7,

所求解集为:

[

[7],

[2,2,3]

]

示例 2:

输入:candidates = [2,3,5],target = 8,

所求解集为:

[

[2,2,2,2],

[2,3,3],

[3,5]

]

class Solution {

    List<List<Integer>> res=new ArrayList<>();

    public List<List<Integer>> combinationSum(int[] candidates, int target) {

        LinkedList<Integer> track = new LinkedList<>();

        if(candidates.length==0) return null;

        backtrace(0,candidates,target,track);

        return res;

    }

    public void backtrace(int start,int[] candidates,int target,LinkedList<Integer> track)

    {

        if(sum(track)==target)res.add(new LinkedList(track));

        for(int i=start;i<candidates.length;i++)

        {

           if(sum(track)>target)continue;

            track.add(candidates[i]);

            backtrace(i,candidates,target,track);

            track.removeLast();

        }

    }

    public int sum(LinkedList<Integer> track)

    {

        int Sum=0;

        for(int item:track) Sum+=item;

        //System.out.println(Sum);

        return Sum;

    }

}



40. 组合总和 II

难度中等

给定一个数组 candidates 和一个目标数 target ,找出 candidates 中所有可以使数字和为 target 的组合。

candidates 中的每个数字在每个组合中只能使用一次。

说明:

所有数字(包括目标数)都是正整数。

解集不能包含重复的组合。 

示例 1:

输入:candidates =[10,1,2,7,6,1,5], target =8,

所求解集为:

[

[1, 7],

[1, 2, 5],

[2, 6],

[1, 1, 6]

]

示例 2:

输入:candidates = [2,5,2,1,2], target = 5,

所求解集为:

[

[1,2,2],

[5]

]

class Solution {

    List<List<Integer>> res=new ArrayList<>();

    public List<List<Integer>> combinationSum2(int[] candidates, int target) {

        LinkedList<Integer> track = new LinkedList<>();

        if(candidates.length==0) return null;

        Arrays.sort(candidates);

        backtrace(0,candidates,target,track);

        return res;

    }

    public void backtrace(int start,int[] candidates,int target,LinkedList<Integer> track)

    {

        if(target<0)return ;

        if(target==0)res.add(new LinkedList(track));

        for(int i=start;i<candidates.length;i++)

        {

           //if(track.contains(candidates[i]))continue;

            if(i>start&&candidates[i]==candidates[i-1])continue;

            track.add(candidates[i]);

            target-=candidates[i];

            backtrace(i+1,candidates,target,track);

            track.removeLast();

            target+=candidates[i];

        }

    }

}



216. 组合总和 III

难度中等

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

说明:

所有数字都是正整数。

解集不能包含重复的组合。 

示例 1:

输入:k= 3,n= 7

输出:[[1,2,4]]

示例 2:

输入:k= 3,n= 9

输出:[[1,2,6], [1,3,5], [2,3,4]]

class Solution {

    List<List<Integer>> res=new ArrayList<>();

    public List<List<Integer>> combinationSum3(int k, int n) {

        if(k==0||k>45)return res;

        LinkedList<Integer> track = new LinkedList<>();

        backtrace(1,track,n,k);

        return res;

    }

    public void backtrace(int loc,LinkedList<Integer> track,int target,int k)

    {

        if(target<0)return;

        else if(track.size()==k&&target==0) res.add(new ArrayList<>(track));

        for(int i=loc;i<10;i++)

        {

            track.add(i);

            target-=i;

            backtrace(i+1,track,target,k);

            target+=i;

            track.removeLast();

        }

    }

}



131. 分割回文串

难度中等

给定一个字符串 s,将 s 分割成一些子串,使每个子串都是回文串。

返回 s 所有可能的分割方案。

示例:

输入:"aab"

输出:

[

["aa","b"],

["a","a","b"]

]

class Solution {

    List<List<String>> res=new ArrayList<>();

    public List<List<String>> partition(String s) {

        LinkedList<String> track=new LinkedList<>();

        int len=s.length();

        if(len<1) return res;

        backtrace(s,track,len,0);

        return res;

    }

    public void backtrace(String s,LinkedList<String> track,int len,int start)

    {

        if(len==start)res.add(new LinkedList(track));

        for(int i=start;i<len;i++)

        {

            if(!check(s,start,i))continue;

            track.add(s.substring(start,i+1));

            backtrace(s,track,len,i+1);

            track.removeLast();

        }

    }

    public boolean check(String s,int left,int right)

    {

        while(left<right)

        {

            if(s.charAt(left)==s.charAt(right))

            {

                left++;

                right--;

            }

            else return false;

        }

        return true;

    }

}



【参考资料】https://mp.weixin.qq.com/s/nMUHqvwzG2LmWA9jMIHwQQ

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 225,124评论 6 523
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 96,453评论 3 404
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 172,386评论 0 368
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 61,136评论 1 301
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 70,142评论 6 400
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 53,593评论 1 315
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 41,958评论 3 429
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 40,944评论 0 279
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 47,477评论 1 324
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 39,512评论 3 346
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 41,639评论 1 355
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 37,227评论 5 351
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 42,971评论 3 340
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 33,397评论 0 25
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 34,550评论 1 277
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 50,203评论 3 381
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 46,713评论 2 366