回溯 八皇后问题 与 0-1背包

八皇后问题描述:

我们有一个 8x8 的棋盘,希望往里放 8 个棋子(皇后),每个棋子所在的行、列、对角线都不能有另一个棋子。你可以看我画的图,第一幅图是满足条件的一种方法,第二幅图是不满足条件的。八皇后问题就是期望找到所有满足这种要求的放棋子方式。

 

 我们把这个问题划分成 8 个阶段,依次将 8 个棋子放到第一行、第二行、第三行……第八行。在放置的过程中,我们不停地检查当前放法,是否满足要求。如果满足,则跳到下一行继续放置棋子;如果不满足,那就再换一种放法,继续尝试。

回溯算法非常适合用递归代码实现,所以,我把八皇后的算法翻译成代码:

public class Queens8 {

    int[] result = new int[8];//全局或成员变量,下标表示行,值表示queen存储在哪一列

    public static void main(String[] args) {
        new Queens8().cal8queens(0);

    }
    public void cal8queens(int row) { // 调用方式:cal8queens(0);
        if (row == 8) { // 8个棋子都放置好了,打印结果
            printQueens(result);
            return; // 8行棋子都放好了,已经没法再往下递归了,所以就return
        }
        for (int column = 0; column < 8; ++column) { // 每一行都有8中放法
            // 这里是分叉路,分岔路下面递归进去执行完后,会退回这里
            // 这里作为最外层的递归起始点,row一直是0,不过是colum一直在移动,寻找各种符合的条件
            // 比如 result[0] = 0,第一个起点,下面有多少种情况,
            // 第二个起点result[0] = 1 还需要用递归判断下面有多少种情况
            if (isOk(row, column)) { // 有些放法不满足要求
                result[row] = column; // 第row行的棋子放到了column列
                System.out.println("row="+row+"\tcolum="+column);
                cal8queens(row+1); // 考察下一行
            }
            System.out.println("------------colum="+column);
        }
    }

    private boolean isOk(int row, int column) {//判断row行column列放置是否合适
        int leftup = column - 1, rightup = column + 1;
        for (int i = row-1; i >= 0; --i) { // 逐行往上考察每一行
            if (result[i] == column) return false; // 第i行的column列有棋子吗?
            if (leftup >= 0) { // 考察左上对角线:第i行leftup列有棋子吗?
                if (result[i] == leftup) return false;
            }
            if (rightup < 8) { // 考察右上对角线:第i行rightup列有棋子吗?
                if (result[i] == rightup) return false;
            }
            --leftup; ++rightup;
        }
        return true;
    }

    private void printQueens(int[] result) { // 打印出一个二维矩阵
        for (int row = 0; row < 8; ++row) {
            for (int column = 0; column < 8; ++column) {
                if (result[row] == column) System.out.print("Q ");
                else System.out.print("* ");
            }
            System.out.println();
        }
        System.out.println();
    }
}

 

0-1背包问题

我们有一个背包,背包总的承载重量是 Wkg。现在我们有 n 个物品,每个物品的重量不等,并且不可分割。我们现在期望选择几件物品,装载到背包中。在不超过背包所能装载重量的前提下,如何让背包中物品的总重量最大?

为什么叫0-1?应为物品是不可分割的,要么装要么不装。

1.确定好最终返回条件

2.转移公式

具体例子:小明有个背包,最大装9kg物品,现在桌上有几个物品,重量(kg)分别是2,2,4,6,3  问,小明包里能装的最大多少kg?

private static  int[] weight = {2,2,4,6,3};  // 物品重量
private static int n = 5; // 物品个数
private static int w = 9; // 背包承受的最大重量
代码如下:
public int maxW = Integer.MIN_VALUE; //存储背包中物品总重量的最大值
// cw表示当前已经装进去的物品的重量和;i表示考察到哪个物品了;
// w背包重量;weight表示每个物品的重量;n表示物品个数
// 假设背包可承受重量9,物品个数5,物品重量存储在数组weight中,那可以这样调用函数:
// f(0, 0, weight, 5, 9)
public void f(int i, int cw, int[] items, int n, int w) {
  if (cw == w || i == n) { // cw==w表示装满了;i==n表示已经考察完所有的物品 
    if (cw > maxW) maxW = cw;
    return;
  }
  f(i+1, cw, items, n, w);    //这里递归只写cw,说明这行代码不装物品
  if (cw + items[i] <= w) {// 已经超过可以背包承受的重量的时候,就不要再装了
    f(i+1,cw + items[i], items, n, w);//这里要装这个物品,但是需要上面的约束条件
  }
}

 回溯可以添加备忘录做优化

优化后如下:

private int maxW = Integer.MIN_VALUE; // 结果放到maxW中
private int[] weight = {2,2,4,6,3};  // 物品重量
private int n = 5; // 物品个数
private int w = 9; // 背包承受的最大重量
private boolean[][] mem = new boolean[5][10]; // 备忘录,默认值false
public void f(int i, int cw) { // 调用f(0, 0)
  if (cw == w || i == n) { // cw==w表示装满了,i==n表示物品都考察完了
    if (cw > maxW) maxW = cw;
    return;
  }
  if (mem[i][cw]) return; // 重复状态
  mem[i][cw] = true; // 记录(i, cw)这个状态
  f(i+1, cw); // 选择不装第i个物品
  if (cw + weight[i] <= w) {
    f(i+1,cw + weight[i]); // 选择装第i个物品
  }
}
View Code

 

posted @ 2020-10-19 17:20  Nucky_yang  阅读(176)  评论(0编辑  收藏  举报