k皇后问题详解java

题目:

  1. 问题描述

2.解题思路重点

首先对于任意一条棋盘上的斜线来说它对应的横坐标x纵坐标y有如下关系:

 

从图中不难看出对于正斜线x-y=(定值),对于反斜线x+y=(定值);

为了找到规律,先从简单的案例入手:

 

如上图在6×6的棋盘中放了4个皇后即m,n,k分别为6,6,4;

则创建三个一维数组x[k + 1],y[k + 1],map[m + 1](这里加1是因为棋盘的行和列是从一开始递增的,这样创建数组可以使数组下标对应到棋盘的行数和列数)。

逐行遍历棋盘,这里就只用第一行举例,若第一个皇后的纵坐标等于1;说明第一个皇后在第一行,根据规则第一行的方格都不能存活;若第一个皇后的纵坐标不等于1说明第一行没有皇后。

接下来看第一个皇后的横坐标,对于皇后所在的列来说:根据规则,第一行一定会有一个方格不能存活,该方格的列数正好对应第一个皇后的横坐标;对于皇后所在的正斜线来说:上图第一个皇后正斜线方程为:x-y=0;这条斜线在第一行时即x = 1时y = 1(说明第一行第一列的方格不能存活),不难发现,对于所有的x,对应的y都可以表示为(x - y + 行数);对于皇后所在的反斜线来说:上图第一个皇后的反斜线方程为x + y = 6;这条斜线在第一行时即x = 1时y = 5(说明第一行第五列的方格不能存活)不难发现对于所有的x,对应的y都可以表示为(x +y- 行数);

这样第一行第一个皇后就判断完了,接着就是遍历剩下的三个皇后;当所有皇后遍历完之后就可一开始判断第二行的死亡格子数了,判断方法和第一行相同,也是从第一个皇后开始。

这里的x[k + 1]用于存储皇后的横坐标,y[k+1]用于存储皇后的纵坐标;而

map[m + 1]用于记录每一行死亡放个数的位置,避免重复判断。

3..源代码

import java.util.Scanner;
public class Main {
    public static void main(String[] args) {
        Scanner sc = new Scanner(System.in);
        int n = sc.nextInt();
        int m = sc.nextInt();
        int k = sc.nextInt();
        int x[] = new int[k + 1];
        int y[] = new int[k + 1];
        int map[] = new int[m + 1];
        for (int i = 1; i < k + 1; i++) {
            y[i] = sc.nextInt();
            x[i] = sc.nextInt();
        }
        int cow = 0, sum = 0;
        for (int i = 1; i <= n; i++) {//这里的i相当于行数,从第一行开始。
            cow = m;//cow是还没判断之前的存活方格数显然为棋盘的列数;
            for (int j = 1; j <= k; j++) {
                int f = i + x[j] - y[j]; //f是正斜线时行数x对应的的列y
                int b = x[j] + y[j] - i;//b是反斜线时行数x对应的的列y

                if(y[j] == i) {
                    cow = 0;
                    break;
                }
                if(map[x[j]] != i) {
                    map[x[j]] = i;
                    cow--;
                }
                if(f > 0 && f <= m && map[f] != i) {
                    map[f] = i;
                    cow--;
                }
                if(b > 0 && b <= m && map[b] != i) {
                    map[b] = i;
                    cow--;
                }
            }
            sum += cow;//sum用来统计一共有多少方格存活。
        }
        System.out.println(sum);
    }
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值