算法设计与分析: 5-20 世界名画陈列馆问题(不重复监视)

该博客介绍了如何设计一个算法解决世界名画陈列馆的问题,即在m×n的陈列馆中,用最少数量的警卫机器人确保每个陈列室被唯一监视。算法采用回溯法来寻找最优解决方案,数据输入包含两个正整数m和n,表示陈列馆的行数和列数。内容引用自王晓东的《计算机算法设计与分析》第三版。

5-20 世界名画陈列馆问题(不重复监视)


问题描述

世界名画陈列馆由 m×nm×n 个排列成矩形阵列的陈列室组成。为了防止名画被盗,需要在陈列室中设置警卫机器人哨位。每个警卫机器人除了监视它所在的陈列室外,还可以监视与 它所在的陈列室相邻的上、下、左、右 4 个陈列室。试设计一个安排警卫机器人哨位的算法, 使得名画陈列馆中每一个陈列室都在警卫机器人的监视之下,并且要求每一个陈列室仅受一 个警卫机器人监视,且所用的警卫机器人数最少。

设计一个算法,计算警卫机器人的最佳哨位安排,使得名画陈列馆中每一个陈列室都仅 受一个警卫机器人监视。且所用的警卫机器人数最少。

数据输入:
第一行有 2 个正整数 m 和 n (1≤m,n≤20)。


Java

package Chapter5HuiSuFa;

import java.util.Scanner;

public class ShiJieMingHuaChenLieGuanBuChongFuJianShi {

    private static int m,n;
    private static int[][] x;
    private static int best;

    public static void main(String[] args){
        Scanner input = new Scanner(System.in);

        while (true){
            n = input.nextInt();
            m = input.nextInt();

            x = new int[n+1][m+1];

            compute();
        }
    }

    private static void compute(){
        boolean ok = false;
        if(n == 1){
            int k = m/3;
            if(m%3 == 1)
                for(int j=0; j<=k; j++)
                    x[1][3*j+1] = 1;
            else{
                if(m%3 == 0) k--;
                for(int j=0; j<=k; j++)
                    x[1][3*j+2] = 1;
            }
            best = k+1;
            ok = true;
        }

        else if(m == 1){
            int k = n/3;
            if(n%3 == 1)
                for(int j=0; j<=k; j++)
                    x[3*j+1][1] = 1;
            else{
                if(n%3 == 0) k--;
                for(int j=0; j<=k; j++)
                    x[3*j+2][1] = 1;
            }
            best = k+1;
            ok = true;
        }

        else if(n==2 && m%2==1){
            int k = m/4;
            if(m%4 == 1){
                for(int j=0; j<=k; j++){
                    x[1][4*j+1] = 1;
                }
                for(int j=0; j<k; j++){
                    x[2][4*j+3] = 1;
                }
            }
            if(m%4 == 3){
                for(int j=0; j<=k; j++){
                    x[1][4*j+1] = 1;
                    x[2][4*j+3] = 1;
                }
            }
            best = (m+1)/2;
            ok = true;
        }

        else if(m==2 && n%2==1){
            int k = n/4;
            if(n%4 == 1){
                for(int j=0; j<=k; j++){
                    x[4*j+1][1] = 1;
                }
                for(int j=0; j<k; j++){
                    x[4*j+3][2] = 1;
                }
            }
            if(n%4 == 3){
                for(int j=0; j<=k; j++){
                    x[4*j+1][1] = 1;
                    x[4*j+3][2] = 1;
                }
            }
            best = (n+1)/2;
            ok = true;
        }

        else if(n==4 && m==4){
            x[1][1] = 1;
            x[1][4] = 1;
            x[4][1] = 1;
            x[4][4] = 1;
            best = 4;
            ok = true;
        }

        if(ok) output();
        else System.out.println("No Solution!");
    }

    private static void output(){
        System.out.println(best);
        for(int i=1; i<=n; i++){
            for(int j=1; j<=m; j++)
                System.out.print(x[i][j]+" ");
            System.out.println();
        }
    }
}

Input & Output

1 1
1
1 

1 2
1
0 1 

1 3
1
0 1 0 

1 4
2
1 0 0 1 

1 5
2
0 1 0 0 1 

1 6
2
0 1 0 0 1 0 

2 1
1
0 
1 

3 1
1
0 
1 
0 

4 1
2
1 
0 
0 
1 

5 1
2
0 
1 
0 
0 
1 

6 1
2
0 
1 
0 
0 
1 
0 

2 2
No Solution!

2 3
2
1 0 0 
0 0 1 

2 4
No Solution!

2 5
3
1 0 0 0 1 
0 0 1 0 0 

2 6
No Solution!

2 7
4
1 0 0 0 1 0 0 
0 0 1 0 0 0 1 

3 2
2
1 0 
0 0 
0 1 

4 2
No Solution!

5 2
3
1 0 
0 0 
0 1 
0 0 
1 0 

6 2
No Solution!

7 2
4
1 0 
0 0 
0 1 
0 0 
1 0 
0 0 
0 1 

3 3
No Solution!

3 4
No Solution!

3 5
No Solution!

3 6
No Solution!

3 7
No Solution!

4 3
No Solution!

5 3
No Solution!

6 3
No Solution!

7 3
No Solution!

4 4
4
1 0 0 1 
0 0 0 0 
0 0 0 0 
1 0 0 1 

4 5
No Solution!

4 6
No Solution!
4 7
No Solution!

5 4
No Solution!

6 4
No Solution!

7 4
No Solution!

5 5
No Solution!

Reference

王晓东《计算机算法设计与分析》(第3版)P186

评论 1
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值