算法设计与分析:世界名画陈列馆问题(可重复监视) (回溯法 分支限界法)

本文主要是介绍算法设计与分析:世界名画陈列馆问题(可重复监视) (回溯法 分支限界法),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

世界名画陈列馆问题

Description:

 世界名画陈列馆由m´n个排列成矩形阵列的陈列室组成。为了防止名画被盗,需要在陈列室中设置警卫机器人哨位。每个警卫机器人除了监视它所在的陈列室外,还可以监视与它所在的陈列室相邻的上、下、左、右个陈列室。试设计一个安排警卫机器人哨位的算法,

使得名画陈列馆中每一个陈列室都在警卫机器人的监视之下,且所用的警卫机器人数最少。

设计一个优先队列式分支限界法,计算警卫机器人的最佳哨位安排,使得名画陈列馆中每一个陈列室都在警卫机器人的监视之下,且所用的警卫机器人数最少。

Input:

第一行有2 个正整数m和n (1≤m,n≤20)

Output:

 将计算出的警卫机器人数及其最佳哨位安排输出。第一行是警卫机器人数;接下来的m行中每行n个数,0 表示无哨位,1 表示哨位。

Sample Input:

4 4
Copy

Sample Output:

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

!!!在各大OJ都提交了,均为超时,不是AC代码。仅作为学习参考。!!!

当y[i][j+1]==1时,以q为根的子树的解,不优于以p为根的子树的解,

当y[i][j+1]==1且y[i][j+2]==1时,以r为根的子树的解,不优于以p为根的子树的解。

搜索时应按照p、q、r或p、r、q的顺序来扩展结点。

剪枝策略:

  1. 放置的机器人个数不会超过n*m/3+1个(按每个机器人仅辐射左右或上下考虑,堆叠这样的小长条可得)。以n*m/3+2为初始最优值,当放置的个数超过当前最优值时,剪去。
  2. (当前最优值ans-当前已放置个数p)*5(最多能增加5个监视点)。如果小于未监视的格点数(n*m-spys),则一定达不到比当前最优值更好的情况,剪去。

 回溯法:

#include <stdio.h>
#include <string.h>
int n,m,f[5][2]={{0,0},{0,1},{0,-1},{1,0},{-1,0}}; //自己本身+上下左右 
int anx[30][30],ans; //最优结果 ans-警卫个数 anx-警卫位置 
int put[30][30],p; //暂时存储 p-警卫个数 put-警卫位置 
int spy[30][30],spys; //spy-被监视的展柜位置 spys-被监视的展柜个数 void puta(int x,int y,int p,int q);
void search(int i,int j)
{if(p>=ans) return;while(i<=n&&spy[i][j]) //已放置的不再被搜索{j++;if(j>m)	i++,j=1; //换行 }if(i>n) //更新答案{ans=p;memcpy(anx, put, sizeof(put)); //把put内容复制给anx return;}//剪枝if((ans-p)*5<=n*m-spys) return;if(i<n) puta(i+1,j,i,j);if(spy[i][j+1]==0) puta(i,j,i,j);if(j<m&&(spy[i][j+1]==0||spy[i][j+2]==0)) puta(i,j+1,i,j);
}void puta(int x,int y,int c,int d)
{put[x][y]=1;p++;for(int i=0;i<5;i++){int xx=x+f[i][0];int yy=y+f[i][1];spy[xx][yy]++;if(spy[xx][yy]==1) spys++;}search(c,d+1);put[x][y] = 0;p--;for(int i=0;i<5;i++){int xx=x+f[i][0];int yy=y+f[i][1];spy[xx][yy]--;if(spy[xx][yy]==0) spys--;}
}int main()
{scanf("%d%d",&n,&m);ans=n*m/3+2;p=0;//设置边界 for(int i=0;i<=n+1;i++)spy[i][0]=spy[i][m+1]=1;for(int i=0;i<=m+1;i++)spy[0][i]=spy[n+1][i]=1;search(1,1);printf("%d\n",ans);for(int i=1;i<=n;i++){for(int j=1;j<=m;j++)printf("%d ",anx[i][j]);printf("\n");}		
}

分支限界法:

#include <stdio.h>
#include <string.h>
#include <queue>
using namespace std;
int n,m,f[5][2]={{0,0},{0,1},{0,-1},{1,0},{-1,0}}; //自己本身+上下左右 
int anx[30][30],ans; //最优结果 ans-警卫个数 anx-警卫位置 struct Node{int pu[30][30];//pu-警卫位置 int spy[30][30]; //spy-被监视的展柜位置int i,j,k,t; //(i,j)为当前坐标 k-警卫数 t-被监视的展柜个数 
};struct cmp //重写比较函数
{bool operator() (Node a, Node b){return a.t > b.t; //小顶堆}
};
//priority_queue<Node> q; //优先队列 
priority_queue<Node, vector<Node>, cmp> q;
Node init(Node node)
{memset(node.pu,0,sizeof(node.pu)); memset(node.spy,0,sizeof(node.spy)); node.i=1;node.j=1;node.k=0;node.t=0;for(int i=0;i<=n+1;i++)node.spy[i][0]=node.spy[i][m+1]=1;for(int i=0;i<=m+1;i++)node.spy[0][i]=node.spy[n+1][i]=1;return node;
}
void puta(Node p,int x,int y)
{Node node;node=init(node);node.i=p.i;node.j=p.j;node.k=p.k+1;node.t=p.t;memcpy(node.pu, p.pu, sizeof(p.pu));memcpy(node.spy, p.spy, sizeof(p.spy));node.pu[x][y]=1;for(int d=0;d<5;d++){int xx=x+f[d][0];int yy=y+f[d][1];node.spy[xx][yy]++;if(node.spy[xx][yy]==1) {node.t++;}}while(node.i<=n&&node.spy[node.i][node.j]) //已放置的不再被搜索{node.j++;if(node.j>m)node.i++,node.j=1; //换行 }q.push(node);return;
}
int main()
{scanf("%d%d",&n,&m);ans=n*m/3+2;Node node;node=init(node);q.push(node);while(!q.empty())//队列非空 {Node p=q.top();q.pop();if(p.t>=n*m){if(p.k<ans){ans=p.k;memcpy(anx, p.pu, sizeof(p.pu)); //把put内容复制给anx }	}else{if(p.i<n) puta(p,p.i+1,p.j);if((p.i==n&&p.j==m)||p.spy[p.i][p.j+1]==0) puta(p,p.i,p.j);if(p.j<m&&(p.spy[p.i][p.j+1]==0||p.spy[p.i][p.j+2]==0)) puta(p,p.i,p.j+1);}}printf("%d\n",ans);for(int i=1;i<=n;i++){for(int j=1;j<=m;j++)printf("%d ",anx[i][j]);printf("\n");}	
}

 

这篇关于算法设计与分析:世界名画陈列馆问题(可重复监视) (回溯法 分支限界法)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!


原文地址:
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.chinasem.cn/article/858641

相关文章

如何解决Druid线程池Cause:java.sql.SQLRecoverableException:IO错误:Socket read timed out的问题

《如何解决Druid线程池Cause:java.sql.SQLRecoverableException:IO错误:Socketreadtimedout的问题》:本文主要介绍解决Druid线程... 目录异常信息触发场景找到版本发布更新的说明从版本更新信息可以看到该默认逻辑已经去除总结异常信息触发场景复

java -jar命令运行 jar包时运行外部依赖jar包的场景分析

《java-jar命令运行jar包时运行外部依赖jar包的场景分析》:本文主要介绍java-jar命令运行jar包时运行外部依赖jar包的场景分析,本文给大家介绍的非常详细,对大家的学习或工作... 目录Java -jar命令运行 jar包时如何运行外部依赖jar包场景:解决:方法一、启动参数添加: -Xb

VS配置好Qt环境之后但无法打开ui界面的问题解决

《VS配置好Qt环境之后但无法打开ui界面的问题解决》本文主要介绍了VS配置好Qt环境之后但无法打开ui界面的问题解决,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要... 目UKeLvb录找到Qt安装目录中designer.UKeLvBexe的路径找到vs中的解决方案资源

Apache 高级配置实战之从连接保持到日志分析的完整指南

《Apache高级配置实战之从连接保持到日志分析的完整指南》本文带你从连接保持优化开始,一路走到访问控制和日志管理,最后用AWStats来分析网站数据,对Apache配置日志分析相关知识感兴趣的朋友... 目录Apache 高级配置实战:从连接保持到日志分析的完整指南前言 一、Apache 连接保持 - 性

MySQL启动报错:InnoDB表空间丢失问题及解决方法

《MySQL启动报错:InnoDB表空间丢失问题及解决方法》在启动MySQL时,遇到了InnoDB:Tablespace5975wasnotfound,该错误表明MySQL在启动过程中无法找到指定的s... 目录mysql 启动报错:InnoDB 表空间丢失问题及解决方法错误分析解决方案1. 启用 inno

Java使用MethodHandle来替代反射,提高性能问题

《Java使用MethodHandle来替代反射,提高性能问题》:本文主要介绍Java使用MethodHandle来替代反射,提高性能问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑... 目录一、认识MethodHandle1、简介2、使用方式3、与反射的区别二、示例1、基本使用2、(重要)

使用C#删除Excel表格中的重复行数据的代码详解

《使用C#删除Excel表格中的重复行数据的代码详解》重复行是指在Excel表格中完全相同的多行数据,删除这些重复行至关重要,因为它们不仅会干扰数据分析,还可能导致错误的决策和结论,所以本文给大家介绍... 目录简介使用工具C# 删除Excel工作表中的重复行语法工作原理实现代码C# 删除指定Excel单元

Linux中的more 和 less区别对比分析

《Linux中的more和less区别对比分析》在Linux/Unix系统中,more和less都是用于分页查看文本文件的命令,但less是more的增强版,功能更强大,:本文主要介绍Linu... 目录1. 基础功能对比2. 常用操作对比less 的操作3. 实际使用示例4. 为什么推荐 less?5.

spring-gateway filters添加自定义过滤器实现流程分析(可插拔)

《spring-gatewayfilters添加自定义过滤器实现流程分析(可插拔)》:本文主要介绍spring-gatewayfilters添加自定义过滤器实现流程分析(可插拔),本文通过实例图... 目录需求背景需求拆解设计流程及作用域逻辑处理代码逻辑需求背景公司要求,通过公司网络代理访问的请求需要做请

电脑蓝牙连不上怎么办? 5 招教你轻松修复Mac蓝牙连接问题的技巧

《电脑蓝牙连不上怎么办?5招教你轻松修复Mac蓝牙连接问题的技巧》蓝牙连接问题是一些Mac用户经常遇到的常见问题之一,在本文章中,我们将提供一些有用的提示和技巧,帮助您解决可能出现的蓝牙连接问... 蓝牙作为一种流行的无线技术,已经成为我们连接各种设备的重要工具。在 MAC 上,你可以根据自己的需求,轻松地