167 - The Sultan's Successors 八皇后(回溯法)

/**
 * 八皇后大水题, 回溯
 * 对数组预处理下,先把所有可能计算出来,存在数组里,也就是把每种情况的八个皇后的坐标记下来。 有92种。
 * 然后再接收输入数据,根据之前处理好的数组,每次计算每种情况的sum,找出最大的输出就行了。
 * 回溯题找出92种情况。。这是最经典最简单的回溯了,也没什么好说的了呃。。
 * 那个判断是否在同一条斜线上的方法也是看《入门经典》里的,横纵坐标相加,或者相减得到的值在左斜或右斜线上是一样的。
 */
#include <cstdio>
#include <cstring>
#include <cmath>
#include <algorithm>
using namespace std;
int chess[8][8], column[8], diagonal1[20], diagonal2[20], ways[100][8];
int curWay, curKind[8];

void dfs(int cur) {
    int i, j;
    if(cur == 8) {
        memcpy(ways[curWay], curKind, 8 * sizeof(int));
        curWay ++;
        return ;
    }
    for(i = 0; i < 8; i ++) {
        if(!column[i] && !diagonal1[cur + i] && !diagonal2[cur - i + 8]) {
            column[i] = diagonal1[cur + i] = diagonal2[cur - i + 8] = true;
            curKind[cur] = i;
            dfs(cur + 1);
            column[i] = diagonal1[cur + i] = diagonal2[cur - i + 8] = false;
        }
    }
}

int main()
{
    int n;
    long long maxSum, sum;
    curWay = 0;
    dfs(0);
    /*   测试下是否成功生成了92种情况
    for(int i = 0; i < 100; i ++) {
        for(int j = 0; j < 8; j ++) {
            printf("%d ", ways[i][j]);

        }   printf("\n");
    }
    */

    scanf("%d", &n);
    while(n --) {
        maxSum = 0;
        for(int i = 0; i < 8; i ++)
            for(int j = 0; j < 8; j ++)
                scanf("%d", &chess[i][j]);
        for(int i = 0; i < 92; i ++) {
            sum = 0;
            for(int j = 0; j < 8; j ++) {
                sum += chess[j][ways[i][j]];
            }
            if(sum > maxSum) maxSum = sum;
        }
        printf("%5lld\n", maxSum);
    }

    return 0;
}


 

你可能感兴趣的:(167 - The Sultan's Successors 八皇后(回溯法))