写在前面

UVA的题需要自己读入一个 \(T\) 组数据,别被样例给迷惑了

Solution

每个格子只有两种填法且 \(n \le 7\),暴力搜索两种填法,开cnt数组统计连接个数。

填一个格子,如果是 "\",格子左上角和右下角的 \(cnt++\),如果是 "/",格子左下角和右上角的 \(cnt++\)。只有更改后的 \(cnt\) 小于等于目标 \(cnt\)才继续向下搜,否则回溯

发现每填一个格子,连接格子左上角的格点的个数就可以被确定,那么只有左上角的格点个数等于目标格点个数或没有要求才继续搜索,否则回溯

如果搜索到第 \(n + 1\) 列时,要额外判断边界第 \(n + 1\) 列的 \(cnt\)是否满足对应的个数

如果搜索到第 \(n\) 行时,要额外判断边界第 \(n + 1\) 行的 \(cnt\) 是否满足对应个数

如何保证无环?不难想到,只有在填 "/" 时才有可能出现环,那么在填 "/" 之前,先判断是否有环

如何处理点的坐标?把行看做十位,把列看做个位就好啦

  • 算法一:考虑可撤销并查集,然而我不会

  • 算法二:发现n很小,每次判断时 \(n^2\) 扫一遍建图,在 \(dfs\) 看看能否从左下角跑到右上角

  • 算法三:很显然算法二很傻逼,直接用并查集维护就好,加完边后判断左下角和右上角是否在同一并查集里,省去 \(dfs\) 的时间,注意清零!

最后输出方案就好啦

Code

我一开始建图跑的,后来也没改(因为懒

/*
Work by: Suzt_ilymics
Knowledge: ??
Time: O(??)
*/
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#define LL long long
#define orz cout<<"lkp AK IOI!"<<endl using namespace std;
const int MAXN = 10;
const int INF = 1;
const int mod = 1; struct edge{
int from, to, nxt;
}e[10100 << 1];
int head[MAXN * MAXN], num_edge = 0; int n;
int go[MAXN][MAXN];
int cnt[MAXN][MAXN], now[MAXN][MAXN];
bool flag = false, Flag = false; int read(){
int s = 0, f = 0;
char ch = getchar();
while(!isdigit(ch)) f |= (ch == '-'), ch = getchar();
while(isdigit(ch)) s = (s << 1) + (s << 3) + ch - '0' , ch = getchar();
return f ? -s : s;
} void add_edge(int from, int to){
e[++num_edge] = (edge){from, to, head[from]}, head[from] = num_edge;
} bool bl(int u, int fa, int end){
for(int i = head[u]; i != -1; i = e[i].nxt){
int v = e[i].to;
if(v == fa) continue;
if(v == end) {
Flag = 1;
return true;
}
bl(v, u, end);
if(Flag) { return true; }
}
return false;
} bool pd(int sx, int sy, int ex, int ey){
memset(head, -1, sizeof(head)); num_edge = 0;
Flag = false;
for(int i = 1; i <= n; ++i){
for(int j = 1; j <= n; ++j){
if(now[i][j] == 0){
add_edge(i * 10 + j, (i + 1) * 10 + j + 1);
add_edge((i + 1) * 10 + j + 1, i * 10 + j);
}
if(now[i][j] == 1){
add_edge((i + 1) * 10 + j, i * 10 + j + 1);
add_edge(i * 10 + j + 1, (i + 1) * 10 + j);
}
}
}
if(bl(sx * 10 + sy, 0, ex * 10 + ey)) return 1;
else return 0;
} void dfs(int posx, int posy){
// cout<<posx<<" "<<posy;
// orz;
if(posx == n && posy == n + 1) {
if( ((cnt[posx + 1][posy] == go[posx + 1][posy]) || (go[posx + 1][posy] == 9)) && ((cnt[posx][posy] == go[posx][posy]) || (go[posx][posy] == 9)) ) flag = 1;
return ;
}// 如果搜到最后一个点,说明已经找到答案。退出
if(posy == n + 1){ //如果这一行搜完了
if((cnt[posx][posy] == go[posx][posy]) || (go[posx][posy] == 9) ) posx++, posy = 1;//换行
else return ;// 如果已经确定的那个数并没有达到目标,返回
}// now[posx][posy] = 0;// 填 "\"
cnt[posx][posy]++, cnt[posx + 1][posy + 1]++;// 对应位置加1 if(cnt[posx][posy] <= go[posx][posy] && cnt[posx + 1][posy + 1] <= go[posx + 1][posy + 1]) {//如果两个对应位置大于目标位置就不向下搜索
if((go[posx][posy] != 9 && go[posx][posy] == cnt[posx][posy]) || (go[posx][posy] == 9)) {//如果已经确定的那个数没有达到目标,停止向下搜索
if((posx != n) || (posx == n && ( (go[posx + 1][posy] != 9 && go[posx + 1][posy] == cnt[posx + 1][posy]) || (go[posx + 1][posy] == 9) ) )){
dfs(posx, posy + 1);//
}
}
}//
if(flag) return ;// 如果找到答案就返回
cnt[posx][posy]--, cnt[posx + 1][posy + 1]--;//回溯 if(pd(posx + 1, posy, posx, posy + 1)){ return ;} now[posx][posy] = 1;// 填 "/"
cnt[posx + 1][posy]++, cnt[posx][posy + 1]++;// 对应位置加1 if(cnt[posx + 1][posy] <= go[posx + 1][posy] && cnt[posx][posy + 1] <= go[posx][posy + 1]) {//如果两个对应位置大于目标位置就不向下搜索
if((go[posx][posy] != 9 && go[posx][posy] == cnt[posx][posy]) || (go[posx][posy] == 9)) {//如果已经确定的那个数没有达到目标,停止向下搜索
if((posx != n) || (posx == n && ( (go[posx + 1][posy] != 9 && go[posx + 1][posy] == cnt[posx + 1][posy]) || (go[posx + 1][posy] == 9) ) )){
dfs(posx, posy + 1);//
}
}
}
if(flag) return ;// 如果找到答案就返回
now[posx][posy] = -1;//回溯
cnt[posx + 1][posy]--, cnt[posx][posy + 1]--;// 回溯
} int main()
{
// freopen("gokigen.in","r",stdin);
// freopen("gokigen.out","w",stdout);
int T;
T = read();
while(T--){
n = read();
memset(now, -1, sizeof(now));
memset(cnt, 0, sizeof(cnt));
memset(go, 0, sizeof(go));
flag = 0;
char ch[10];
for(int i = 1; i <= n + 1; ++i){
cin>>(ch + 1);
for(int j = 1; j <= n + 1; ++j){
if(isdigit(ch[j])) go[i][j] = ch[j] - '0';
else go[i][j] = 9;
}
} dfs(1, 1);
// for(int i = 1; i <= n + 1; ++i){
// for(int j = 1; j <= n + 1; ++j){
// cout<<go[i][j]<<" ";
// }
// cout<<"\n";
// }
// cout<<"\n";
//
// for(int i = 1; i <= n + 1; ++i){
// for(int j = 1; j <= n + 1; ++j){
// cout<<cnt[i][j]<<" ";
// }
// cout<<"\n";
// }
// cout<<"\n"; for(int i = 1; i <= n; ++i){
for(int j = 1; j <= n; ++j){
if(now[i][j] == 1) cout<<"/";
else if(now[i][j] == 0) cout<<"\\";
else cout<<"s";
}
cout<<"\n";
}
}
return 0;
}

UVA11694 Gokigen Naname题解的更多相关文章

  1. 题解 UVA11694 【Gokigen Naname谜题 Gokigen Naname】

    题目 题解 考场上连暴力都不会打的码农题,深搜是真的难 /kk 前置问题 怎么输出"\" cout<<"\\"; 2.怎么处理不在一个环里,可以考虑 ...

  2. Uva 11694 Gokigen Naname

    基本思路是Dfs: 1. 一个一个格子摆放,以每个各自的左上角的点为基准点代表格子,比如(0,0)代表(0,0)(0,1)(1,0)(1,1)组成的格子,(0,1)代表(0,1)(0,2)(1,1), ...

  3. UVA11694-Gokigen Naname(DFS进阶)

    Problem UVA11694-Gokigen Naname Accept: 76   Submit: 586Time Limit: 10000 mSec Problem Description I ...

  4. 2016 华南师大ACM校赛 SCNUCPC 非官方题解

    我要举报本次校赛出题人的消极出题!!! 官方题解请戳:http://3.scnuacm2015.sinaapp.com/?p=89(其实就是一堆代码没有题解) A. 树链剖分数据结构板题 题目大意:我 ...

  5. noip2016十连测题解

    以下代码为了阅读方便,省去以下头文件: #include <iostream> #include <stdio.h> #include <math.h> #incl ...

  6. BZOJ-2561-最小生成树 题解(最小割)

    2561: 最小生成树(题解) Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 1628  Solved: 786 传送门:http://www.lyd ...

  7. Codeforces Round #353 (Div. 2) ABCDE 题解 python

    Problems     # Name     A Infinite Sequence standard input/output 1 s, 256 MB    x3509 B Restoring P ...

  8. 哈尔滨理工大学ACM全国邀请赛(网络同步赛)题解

    题目链接 提交连接:http://acm-software.hrbust.edu.cn/problemset.php?page=5 1470-1482 只做出来四道比较水的题目,还需要加强中等题的训练 ...

  9. 2016ACM青岛区域赛题解

    A.Relic Discovery_hdu5982 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Jav ...

随机推荐

  1. bladex从blade-dev.yaml 读取配置信息

    blade-dev.yaml配置======nacos文件配置 #sap配置 sap: api: read: url: http://read.xxxxxxxx.com.cn port: 80 use ...

  2. 技术选型关于redis客户端选择

    redis作为分布式缓存框架的首选  相信已经毋庸置疑.能高效.合理的使用好它  必定能提升系统的可用性,高性能.高吞吐量的保障.但选择一个客户端,充分发挥它的能力,就是一个选型问题.现在市场上能选择 ...

  3. linux下 shell时间处理

    一.hour #获取当前时间年月日时分秒current_create_time=`date +"%Y-%m-%d %H:%M:%S"` echo $current_create_t ...

  4. python之json、pickle模块

    一.json模块 之前我们学习过用eval内置方法可以将一个字符串转成python对象,不过,eval方法是有局限性的,对于普通的数据类型,json.loads和eval都能用,但遇到特殊类型的时候, ...

  5. java:原子类的CAS

    当一个处理器想要更新某个变量的值时,向总线发出LOCK#信号,此时其他处理器的对该变量的操作请求将被阻塞,发出锁定信号的处理器将独占共享内存,于是更新就是原子性的了. 1.compareAndSet- ...

  6. js 的关键字

    1.get / set var test = { _Name: "Limei", _Age: 20, get name() { return this._Name;}, set a ...

  7. dede织梦技巧:教你彻底解决dede按权重排序的问题(转)

    dede排序对网站来说一直存在问题,默认是按照最新发布时间排序.这样排序有个问题,一旦更新之后即被视为最新发布,于是原本做好的排序瞬间就乱了. 这种时候,按权重排序是个很好的选择,但按权重排序到处存在 ...

  8. day116:MoFang:显示背包解锁/未解锁格子数&显示背包的道具物品&背包解锁

    目录 1.显示背包的已解锁/未解锁格子数 2.显示背包中的道具物品 3.用户购买道具的时候,判断背包存储是否达到上限 4.道具也可以使用积分购买 5.在商城界面根据金额/积分显示不同商品 6.背包解锁 ...

  9. Java NIO 文件通道 FileChannel 用法

    FileChannel 提供了一种通过通道来访问文件的方式,它可以通过带参数 position(int) 方法定位到文件的任意位置开始进行操作,还能够将文件映射到直接内存,提高大文件的访问效率.本文将 ...

  10. 10步写了个Django网站,正经网站···

      Django做网站只要10步,真的只有10步,不信?咱们来数数--   今天主要讲解用Pycharm编辑器搭建网站,网站功能是 实现在局域网中快速传递大文件! 比如:同事要给你个1G的文件,你丢一 ...