100722A
这道题抄了答案:
思路:旋转,其实只用旋转四次,因为在换行的过程中旋转其实是没有意义的,因为行列只不过转了个角度。然后主要的是行列的交换,这里我很头疼,写了个盲目搜索,当然wa掉了
问了问某位同志,是这样做的,很巧妙:其实这是一个全排列,一个三个数的全排列套了三个数的全排列,然后该怎么做呢?这里运用了一个巧妙的方法:range
range很巧妙,分两种情况:当我选完了三列时,这时我需要新的一列,那么我返回(0,8),意思是下一个列可以从(0,8)之间没用过的任选。第二呢,是我正在选三列中的一列,这时
我看看上一列我选的什么,这次我选的只能是(上一列设为x)[x/3*3,x/3*3+2]举个例子:(0,1,2) (3,4,5)(6,7,8)其实就是这三个块,他们每三个数必须绑在一起
然后我搜完了列之后搜行(搜行的时候我犯了个错误,对应残缺的数独下标没想清楚,debug了很长时间,下次应该想清楚)然后就是重复同样的过程,只不过要加上置换,
置换需要用一个额外的数组记录是那一列的时候替换的,因为会导致还原的时候不知道哪些要还原,然后就好了。
#include<iostream>
#include<cstring>
#include<cstdio> using namespace std; typedef pair<int,int> pii; string s;
int used_col[],used_row[],place[];
int pos_row[],pos_col[],place_flag[];
int b[][],c[][]; void init()
{
memset(place_flag,-,sizeof(place_flag));
memset(place,-,sizeof(place));
memset(pos_row,-,sizeof(pos_row));
memset(pos_col,-,sizeof(pos_col));
memset(used_row,,sizeof(used_row));
memset(used_col,,sizeof(used_col));
} pii get_range(int pos,int f)
{
if(pos%==) return pii(,);
int head=;
if(!f) head=pos_row[pos-]/*;
else head=pos_col[pos-]/*;
return pii(head,head+);
} void rotate()
{
int a[][];memset(a,,sizeof(a));
for(int i=;i<;i++)
for(int j=;j<;j++) a[-j-][i]=b[i][j];
for(int i=;i<;i++)
for(int j=;j<;j++) b[i][j]=a[i][j];
} bool go_col(int d)
{
if(d==) return true;
pii range=get_range(d,);
for(int i=range.first;i<=range.second;i++) if(!used_col[i])
{
used_col[i]=;
pos_col[d]=i;
bool flag=false;
for(int j=;j<;j++)
{
int x=b[pos_row[j]][pos_col[d]],y=c[j][d];
if(y==) continue;
if(place[x]!=-&&place[x]!=y) {flag=true;break;}
if(place[x]==-)
{
place[x]=y;
place_flag[x]=d;
}
}
if(!flag) if(go_col(d+)) return true;
for(int j=;j<;j++)
{
int x=b[pos_row[j]][pos_col[d]];
if(place_flag[x]==d) {place_flag[x]=-;place[x]=-;}
}
used_col[i]=;
pos_col[d]=-;
}
return false;
} bool go_row(int d)//行
{
if(d==) return go_col();
pii range=get_range(d,);
for(int i=range.first;i<=range.second;i++)
{
if(!used_row[i])
{
used_row[i]=;
pos_row[d]=i;
if(go_row(d+)) return true;
used_row[i]=;
pos_row[d]=-;
}
}
return false;
} bool work()
{
for(int i=;i<;i++)
{
init();
rotate();
if(go_row()) return true;
}
return false;
} int main()
{
int T;scanf("%d",&T);
while(T--)
{
for(int i=;i<;i++)
{
cin>>s;
for(int j=;j<;j++)
{
b[i][j]=s[j]-'';
}
}
for(int i=;i<;i++)
{
cin>>s;
for(int j=;j<;j++)
{
c[i][j]=s[j]-'';
}
}
if(work()) printf("Yes\n"); else printf("No\n");
}
return ;
}
100722A的更多相关文章
随机推荐
- AC日记——画矩形 1.5 42
42:画矩形 总时间限制: 1000ms 内存限制: 65536kB 描述 根据参数,画出矩形. 输入 输入一行,包括四个参数:前两个参数为整数,依次代表矩形的高和宽(高不少于3行不多于10行,宽 ...
- Deploying an Internet Information Services-Hosted WCF Service
Deploying an Internet Information Services-Hosted WCF Service .NET Framework 4 Other Versions .NET ...
- [No000060]冷读热读:读书九问
兵无常势,水无常形,读书亦无法.彼之砒霜,我之佳肴.然读书无法却有道.你我都是使用同一颗大脑在读书.这颗大脑受制于那千千万万年以来,星辰起落,狩猎采集,演化大道. Q1:读物如何分级? 坏书.可用的书 ...
- 渗透攻防工具篇-后渗透阶段的Meterpreter
作者:坏蛋链接:https://zhuanlan.zhihu.com/p/23677530来源:知乎著作权归作者所有.商业转载请联系作者获得授权,非商业转载请注明出处. 前言 熟悉Metasploit ...
- uva216 Getting in Line
Computer networking requires that the computers in the network be linked. This problem considers a \ ...
- 理解java的finalize
基本预备相关知识 1 java的GC只负责内存相关的清理,所有其它资源的清理必须由程序员手工完成.要不然会引起资源泄露,有可能导致程序崩溃. 2 调用GC并不保证GC实际执行. 3 finalize抛 ...
- Codevs 1230 STL万岁。。 。
题目描述 Description 给出n个正整数,然后有m个询问,每个询问一个整数,询问该整数是否在n个正整数中出现过. 输入描述 Input Description 第一行两个整数 n 和m. 第二 ...
- iOS SDWebImage使用详解
这个类库提供一个UIImageView类别以支持加载来自网络的远程图片.具有缓存管理.异步下载.同一个URL下载次数控制和优化等特征. 使用示范的代码:UITableView使用UIImageView ...
- 傅盛:如何快慢“炼”金山?(转)
原文地址:http://www.huxiu.com/article/16052/1.html 一直以来,金山都不是一家"大公司",从前不是,现在也不是. 能够掰着指头数完腾讯六大事 ...
- jQuery使用.on()无法绑定hover
发现好像没有hover这个事件,jQuery的hover事件是一个封装,hover算不得一个事件.他只是将mouseover和mouseout合并了用mouseover和mouseout两个配合效果好 ...