首页 > 其他 > 详细

列举常见的递归问题

时间:2020-09-27 08:24:11      阅读:32      评论:0      收藏:0      [点我收藏+]

因为今天我重新看了递归的教学视频,但发现好像很多都很容易忘,所以我想把所有常见的递归算法题给整理一下。

1.全排列

输入:"aac"
输出:[aac,aac , aca, aca,caa]

思路就是可以选择每个下标的字母作为开头。

public static List<String> getAll(String str) {
        if(str.equals("") || str == null){
            return null;
        }
        List<String> list = new ArrayList<>();
        ArrayList<Character> set = new ArrayList<Character>();
        for (Character character : str.toCharArray()) {
            set.add(character);
        }
        process(set,"",list);
        return list;
    }
public static void process(ArrayList<Character> set, String path, List<String> list){
        if(set.isEmpty()){
            list.add(path);
            return;
        }
        for (int index = 0; index < set.size(); index++) {
                String pick = path+set.get(index);
                ArrayList<Character> next = new ArrayList<>(set);
                next.remove(index);
                process2(next, pick, list);
            }
        }
}

2.全排列,不要有重复

这里要注意判断是否选过的HashSet<Character> picks,这个点我想了好久才想通,因为在for循环里面,其实是在循环每个位置作为开头,所以在循环的时候,比如当我们选了‘a’之后,我们就不能再选‘a’作为开头字母了,所以我觉得从循环这个点去理解HashSet<Character> picks会比较好。

public static List<String> getAll(String str) {
        if(str.equals("") || str == null){
            return null;
        }
        List<String> list = new ArrayList<>();
        ArrayList<Character> set = new ArrayList<Character>();
        for (Character character : str.toCharArray()) {
            set.add(character);
        }
        process1(set,"",list);
        return list;
    }
public static void process1(ArrayList<Character> set, String path, List<String> list){
        if(set.isEmpty()){
            list.add(path);
            return;
        }
        HashSet<Character> picks = new HashSet<>();
        for (int index = 0; index < set.size(); index++) {
            if (!picks.contains(set.get(index))) {
                picks.add(set.get(index));
                String pick = path+set.get(index);
                ArrayList<Character> next = new ArrayList<>(set);
                next.remove(index);
                process1(next, pick, list);
            }
        }
    }
}

3.解码

输入: "12"
输出: 2
解释: 它可以解码为 "AB"(1 2)或者 "L"(12)。

这个其实跟全排列的思路是一样的,就是我们去遍历给的数字字符串,我们可以判断是否选当前字符,还是选择当前字符加下一位字符。其实我有点想太多了,其实就是递归的时候加上一些if语句。

public static int numDecodings(String s) {
        if(s == null || s.equals("")) return 0;
        char[] ch = s.toCharArray();
        return process(ch, 0);
        
    }
public static int process(char[] ch, int index){  
                if(index == ch.length){
                        return 1;
                }
                if(ch[index] == ‘0‘){
                        return 0;
                }
                if(ch[index] == ‘1‘){
                        int res = process(ch,index+1);
                        if(index+1 < ch.length){
                             res+= process(ch,index+2);
                        }
                        return res;
                }
                if(ch[index] == ‘2‘){
                      int res = process(ch.index+1);
                      if(index+1 < ch.length && ch[index] <‘6‘){
                           res += process(ch,index+2);
                      }
                      return res;
               }
               return process(ch,index+1);
         }
}       

4.背包问题

给定两个长度为N的数组weights和values,weights[i]和values[i]分别表示i号物品的重量和价值,
给定一个正数bag,表示一个载重bag的袋子,你装的物品不能超过这个重量,返回你能装下最大价值是多少

这个也是一个很经典的题目

public static int maxValue1(int[] c, int[] p, int bag) {
        return process1(c, p, 0, 0, bag);
    }
public static int process1(int[] weights, int[] values, int i, int alreadyweight, int bag) {
        if (alreadyweight > bag) {
            return 0;
        }
        if (i == weights.length) {
            return 0;
        }
        return Math.max(
                
                process1(weights, values, i + 1, alreadyweight, bag),
                
                values[i] + process1(weights, values, i + 1, alreadyweight + weights[i], bag));
}

5.选牌

两位绝顶聪明的选手做选牌的游戏,桌子上放着代表这价值的卡牌,轮流拿牌,但是只能拿最左边或者最右边的牌。
public static int win(int[] arr){
        if (arr == null || arr.length == 0) {
            return 0;
        }
        return Math.max(xianshou(arr, 0, arr.length-1), 
                houshou(arr, 0, arr.length-1));
    }
    public static int xianshou(int[] arr, int L ,int R){
        if(L == R){
            return arr[L];
        }
        return Math.max(arr[L]+houshou(arr, L+1, R),
                        arr[R]+houshou(arr, L, R-1));
    }
    public static int houshou(int[] arr,int L ,int R){
        if(L == R){
            return 0;
        }
        return Math.min(xianshou(arr, L-1, R), 
                xianshou(arr, L, R-1));
    }

6.N皇后问题

public static int num1(int n) {
        if(n < 1) return 0;
        int[] record = new int[n];
        return process(n, 0, record);
    }
    
    public static int process(int n , int i ,int[] record){
        if(i == n){
            return 1;
        }
        int res = 0;
        for (int j = 0; j < n; j++) {
            if(isValid(record,i ,j)){
                record[i] = j;
                res +=process(n, i+1, record);
            }
        }
        return res;
    }
    public static boolean isValid(int[] record, int i ,int j){
        for (int k = 0; k < i; k++) {
            if(record[k] == j || Math.abs(k-i) == Math.abs(record[k] - j)  ){
                return false;
            }
        }
        
        return true;
    }
    public static int num2(int n) {
        if(n<2 || n>32) return 0;
        int limit = n == 32 ? -1 : (1<<n) -1;
        return process(limit,0,0,0);
    }
    public static int process(int limit,
                                int colLim,
                                int leftDaLim,
                                int rightDaLim){
        if(colLim == limit){
            return 1;
        }
        int positon = limit & ~(colLim | leftDaLim | rightDaLim); 
        int mostRightPos = 0;
        int res = 0;
        while(positon != 0){
            mostRightPos = positon & (~positon +1);
            positon = positon - mostRightPos;
            res += process(limit, 
                    colLim | mostRightPos, 
                    (leftDaLim| mostRightPos )<<1, 
                    (rightDaLim | mostRightPos )>>>1);
        }
        return res;
    }

判断是否处于相同的斜对角线,只要知道两个点的行列,

比如点1(a,b)和点2(b,c),|a-c|==|b-d|

7.机器人走路

给定一个数值,机器人每次只能前进一步和后退一步,指定机器人只能走k步,返回机器人能走到指定位置的路径数。
输入:n=
输出:
public static int walk(int N, int s, int e, int k){
        return process2(N,s,e,k);
    }
    public static int process(int N ,int cur, int e, int rest){
        if(rest == 0){
            return rest == e ? 1 : 0 ;
        }
        if (cur == 1) {
            return process(N, cur+1, e, rest-1);
        }
        if (cur == N) {
            return process(N, N-1, e, rest-1);
        }
        return process(N, cur+1, e, rest-1)+process(N, cur-1, e, rest-1);
    }
    public static int process2(int N ,int cur, int e, int rest){
        int[][] dp = new int[rest+1][N+1];
        //解决第一行的dp表
        dp[0][e] = 1;
        //遍历行
        for (int i = 1; i <= rest; i++) {
            //遍历列
            for (int j = 1; j <= N; j++) {
                if(j == 1){
                    dp[i][j] = dp[i-1][j+1]; 
                }else if (j == N){
                    dp[i][j] = dp[i-1][j-1];
                }else {
                    dp[i][j] = dp[i-1][j-1] + dp[i-1][j+1]; 
                }
            }
        }
        return dp[e][rest];
    }

 

列举常见的递归问题

原文:https://www.cnblogs.com/carryup/p/13737429.html

(0)
(0)
   
举报
评论 一句话评论(0
关于我们 - 联系我们 - 留言反馈 - 联系我们:wmxa8@hotmail.com
© 2014 bubuko.com 版权所有
打开技术之扣,分享程序人生!