题目大意

  有一个由火柴棍组成的边长为n的正方形网格,每条边有n根火柴,共2n(n+1)根火柴。从上至下,从左到右给每个火柴编号,现在拿走一些火柴,问在剩下的后拆当中ongoing,至少还要拿走多少根火柴才能破坏掉所有的正方形?



  虽然本题的数据规模不大,但是却有多种选择火柴棍的方法,导致如果直接爆搜的话会炸,因此考虑采用IDA*算法。

  首先来看,我们该如何存储这个火柴棍图?

  个人比较喜欢把这样一个火柴棍图转化成这样的数组形式,存为square[][]

   0 1 0 1 0 1 0

   1 0 1 0 1 0 1

   0 1 0 1 0 1 0

   1 0 1 0 1 0 1

   0 1 0 1 0 1 0

   1 0 1 0 1 0 1

   0 1 0 1 0 1 0

   其中,0代表火柴棍交界的地方,1代表火柴棍。

  先给所有火柴棍编上号。在a上遍历,按序命名,用一个map<pair<int,int>,int> name进行映射

  之后,给每一个正方形编一个号,在用一个vector存一下每个正方形的所有边,便于进行搜索时快速确定本次该删除哪一条边。在存储边的同时,还要记录下当前这个正方形的原大小(即火柴棍全部摆满)和现在的大小(删除火柴棍后的数量),比较这两个的大小即可确定现在这个正方形是否被破坏。存储过程用函数实现

点击查看代码
void get(int x,int y,int len,int ccnt) {//搜正方形,并把组成正方形的每个火柴添加上当前的正方形
int tx = x,ty = y;
for(ty = y; ty <= 2 * n + 1; ty++){//向右走
if(cnt == len){cnt = 0;break;}
if(square[tx][ty]){
cnt++;
match[ccnt].push_back(name[make_pair(tx,ty)]);
}
}
for(tx = x; tx <= 2 * n + 1; tx++){//向下走
if(cnt == len){cnt = 0;break;}
if(square[tx][ty]){
cnt++;
match[ccnt].push_back(name[make_pair(tx,ty)]);
}
}
for(ty; ty >= 1; ty--){//向左走
if(cnt == len){cnt = 0;break;}
if(square[tx][ty]){
cnt++;
match[ccnt].push_back(name[make_pair(tx,ty)]);
}
}
for(tx; tx >= 1; tx--){
if(cnt == len){cnt = 0;break;}
if(square[tx][ty]){
cnt++;
match[ccnt].push_back(name[make_pair(tx,ty)]);
}
}
}

  IDA*的关键就在于估价函数,这个题该如何来写估价函数呢?
  理想状况下,破坏一个正方形,可以顺带把这个正方形周围的正方形全部破坏(这就是估价函数和搜索函数check函数不同的原因,tmp当然就是来记录最理想状况的),当然,只是理想状况。如果按照这种方式进行破坏也不能在规定时间内破坏完毕,那这样的方案一定不可行,直接剪枝。
搜索时,以没有被破坏的正方形为对象,遍历它的每一条边,然后挨个搜索即可

  在实现当中,可以看见我们使用了memcpy这个函数,它可以将一个数组内的值复制到另一个数组里面,挺方便的

点击查看代码
#include<bits/stdc++.h>
using namespace std;
const int MAXN = 22;
int t,n;
bool square[MAXN][MAXN],use[MAXN * MAXN];//vis表示正方形i是否被破坏 ,use表示该火柴棍是否被使用
int cnt,now,del;
map<pair<int,int>,int> name;
bool found;
bool tmp[MAXN * MAXN];
vector<int> match[MAXN * MAXN];//编号i的正方形由哪些火柴棍组成
void get(int x,int y,int len,int ccnt) {//搜正方形,并把组成正方形的每个火柴添加上当前的正方形
int tx = x,ty = y;
for(ty = y; ty <= 2 * n + 1; ty++){//向右走
if(cnt == len){cnt = 0;break;}
if(square[tx][ty]){
cnt++;
match[ccnt].push_back(name[make_pair(tx,ty)]);
}
}
for(tx = x; tx <= 2 * n + 1; tx++){//向下走
if(cnt == len){cnt = 0;break;}
if(square[tx][ty]){
cnt++;
match[ccnt].push_back(name[make_pair(tx,ty)]);
}
}
for(ty; ty >= 1; ty--){//向左走
if(cnt == len){cnt = 0;break;}
if(square[tx][ty]){
cnt++;
match[ccnt].push_back(name[make_pair(tx,ty)]);
}
}
for(tx; tx >= 1; tx--){
if(cnt == len){cnt = 0;break;}
if(square[tx][ty]){
cnt++;
match[ccnt].push_back(name[make_pair(tx,ty)]);
}
}
}
bool _check(int i){
for(int j=0;j<match[i].size();++j){
if(use[match[i][j]])
return 0;
}
return 1;//没被破坏
}
bool check(int i){
for(int j=0;j<match[i].size();++j){
if(tmp[match[i][j]])
return 0;
}
return 1;
}
void init(){
name.clear();
for(int i = 1; i < MAXN*MAXN; i++)match[i].clear();
memset(square,0,sizeof square);
memset(use,0,sizeof use);
memset(tmp,0,sizeof tmp);
}
int estimate(){
int sum = 0;
memcpy(tmp,use,sizeof(use));
for(int i = 1; i <= now; i++){
if(check(i)){
sum++;
for(int j=0;j<match[i].size();++j)
tmp[match[i][j]]=1;
}
}
return sum;
}
void IDAstar(int x,int maxd){
if(x+estimate()>maxd)return;
for(int i=1;i<=now;++i){
if(_check(i)){
for(int j=0;j<match[i].size();++j){
use[match[i][j]]=1;
//v.push_back(id[i][j]);
IDAstar(x+1,maxd);
//v.pop_back();
use[match[i][j]]=0;
if(found)return;
}
return;
}
}
found=1;
return;
}
int main(){
cin >> t;
while(t--){
init();
cin >> n;
bool flag = 0;
cnt = now = 0;
for(int i = 1; i <= 2 * n + 1; i++){//构造正方形图并给火柴棍编号
if(i % 2 == 0)flag = 1;
for(int j = 1; j <= 2 * n + 1; j++){
square[i][j] = flag;
if(flag == 1){
cnt++;
name[make_pair(i,j)] = cnt;
} flag = !flag;
}
}
cnt = 0; for(int i = 1; i <= 2 * n + 1; i++){//寻找正方形
if(i % 2 == 0)continue;
for(int j = 1; j <= 2 * n + 1; j++){
if(!square[i][j])continue;
for(int len = 1; len <= n; len++){
if((i + 2 * (len - 1) + 1 > 2 * n + 1) || (j + 2 * (len - 1) + 1 > 2 * n + 1))break;
now++;
get(i,j,len,now);
} }
}
cin >> del;
int a;
for(int i = 1; i <= del; i++){
cin >> a;
use[a] = 1;
}
for(int i = estimate();;i++){
found = 0;
IDAstar(0,i);
if(found){
cout << i << "\n";
break;
}
}
}
}

破坏正方形UVA1603的更多相关文章

  1. 7-15 Square Destroyer 破坏正方形 uva1603

    先是处理所有的正方形 从边长为1开始 将其边存好 满边存好 然后不断扫描正方形  并且进行拆除  直到拆完或者 步数小于等于9(启发方程  因为n小于等于5  九次足以将所有的拆完) 代码实施有很多细 ...

  2. UVa 1603 破坏正方形

    https://vjudge.net/problem/UVA-1603 题意:有一个火柴棍组成的正方形网格,计算至少要拿走多少根火柴才能破坏所有正方形. 思路:从边长为1的正方形开始遍历,将正方形的边 ...

  3. IDA*

    模拟退火 基本思路(Main Thoughts): IDA*是一种优秀的搜索法,在一般的实际问题中,它比普通的搜索更快. 通过迭代加深和估价函数剪枝来搜索. 通常处理没有层数上界或上界很多大的搜索. ...

  4. [LeetCode] Matchsticks to Square 火柴棍组成正方形

    Remember the story of Little Match Girl? By now, you know exactly what matchsticks the little match ...

  5. [LeetCode] Maximal Square 最大正方形

    Given a 2D binary matrix filled with 0's and 1's, find the largest square containing all 1's and ret ...

  6. 【纯css】左图右文列表,左图外框宽度占一定百分比的正方形,右上下固定,右中自动响应高度。支持不规则图片。

    查看演示 <!DOCTYPE html> <html> <head lang="en"> <meta charset="UTF- ...

  7. Mint Linux 安装 DotnetCore 遭遇无法修正错误,因为您要求某些软件包保持现状,就是它们破坏了软件包间的依赖关系

    evlon@evlon-ThinkPad-T530 ~ $ apt install dotnet-dev-1.0.0-preview2-003121 正在读取软件包列表... 完成 正在分析软件包的依 ...

  8. BZOJ1047: [HAOI2007]理想的正方形 [单调队列]

    1047: [HAOI2007]理想的正方形 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 2857  Solved: 1560[Submit][St ...

  9. [译]JavaScript源码转换:非破坏式与再生式

    原文:http://ariya.ofilabs.com/2013/06/javascript-source-transformation-non-destructive-vs-regenerative ...

随机推荐

  1. mybatis入门,CRUD,万能Map,模糊查询

    第一个Mybatis程序 核心配置文件mybatis-config.xml <?xml version="1.0" encoding="UTF-8" ?& ...

  2. python之装饰器补充与递归函数与二分查找

    目录 多层装饰器 有参装饰器 递归函数 基本演示 斐波那契数列 总结 小拓展 算法之二分法 简介 举例 总结 多层装饰器 我们已经知道了语法糖的作用是将装饰对象自动装饰到装饰器中,一个语法糖的应用我们 ...

  3. DCM:一个能够改善所有应用数据交互场景的中间件新秀

    摘要:几乎所有涉及应用数据交互的场景都可以通过DCM来改善应用结构,提升开发与计算效率. 本文分享自华为云社区<DCM:中间件家族迎来新成员>,作者: 石臻臻的杂货铺. DCM是什么 现代 ...

  4. 第6组 Beta冲刺 总结

    目录 1. 基本情况 2. 思考与总结 2.1. 设想和目标 2. 计划 3. 资源 4. 变更管理 5. 设计/实现 6. 测试/发布 7. 团队的角色,管理,合作 8. 总结 3. 敏捷开发 1. ...

  5. 第一次的ssm整合

    数据库表 导入依赖 <dependencies> <dependency> <groupId>javax.servlet</groupId> <a ...

  6. Cpp的赋值和变量说明

    一命名方式: 1.关键字不能作为变量名 int int;是错误的电脑会提示为非法取名 上面的示例是错误示范,而错误提示告诉了为什么错了记住这错误提示了: 2.的二个知识点: 变量名是分大小写的: in ...

  7. 解决python 导入selenium 库后自动化运行成功但是报错问题

    本章节开始进入自动化的基础教学了,首先我们要对我们的工具有一定的熟练使用程度,做自动化常用的工具一个是搭建 RobotFramework自动化框架,另外一个便是我们最常用的python 工作原理是比较 ...

  8. 2021.06.05【NOIP提高B组】模拟 总结

    T1 题意:给你一个 \(n\) 个点 \(n\) 条边的有向图, 求每个店经过 \(K\) 条边后的边权和.最小边权 \(K\le 10^{10}\) 考试时:一直想着环,结果一直不知道怎么做 正解 ...

  9. 互联网公司目标管理OKR和绩效考核的误区

    最近看了一篇关于「谷歌放弃OKR,转向全新的GRAD系统」的文章,我转到了研发效能DevOps的微信群里,结果引起了大家热烈的讨论,正好我们也在使用 OKR,所以也来谈谈我的理解以及我们应用起来的实际 ...

  10. ubuntu使用postfix和AWS-SES发送邮件

    在日常开发中,邮件发送是个比较常见的场景.因此出现了很多相关的软件和服务,各大云厂商也推出自己的邮件服务.今天笔者就像大家介绍一种常见的组合,AWS的邮件服务 SES 与邮件服务器 postfix 的 ...