【NOIP2004】虫食算
Description
所谓虫食算,就是原先的算式中有一部分被虫子啃掉了,需要我们根据剩下的数字来判定被啃掉的字母。来看一个简单的例子:
43#9865#045
+. 8468#6633
44445509678
其中#号代表被虫子啃掉的数字。根据算式,我们很容易判断:第一行的两个数字分别是5和3,第二行的数字是5。
现在,我们对问题做两个限制:首先,我们只考虑加法的虫食算。这里的加法是N进制加法,算式中三个数都有N位,允许有前导的0。其次,虫子把所有的数都啃光了,我们只知道哪些数字是相同的,我们将相同的数字用相同的字母表示,不同的数字用不同的字母表示。如果这个算式是N进制的,我们就取英文字母表午的前N个大写字母来表示这个算式中的0到N-1这N个不同的数字:但是这N个字母并不一定顺序地代表0到N-1)。输入数据保证N个字母分别至少出现一次。
. BADC
+CBDA
. DCCC
上面的算式是一个4进制的算式。很显然,我们只要让ABCD分别代表0123,便可以让这个式子成立了。你的任务是,对于给定的N进制加法算式,求出N个不同的字母分别代表的数字,使得该加法算式成立。输入数据保证有且仅有一组解
解题报告
搜索剪枝,实际上状态很少,考虑怎么样的搜索顺序使得状态最少,肯定是按位枚举,把确定的字母标记一下,并且标记哪些数字被用过,产生矛盾就停止搜索,注意回溯
另外就是判断每一位的等式是否成立,注意进位的问题,递归时记录即可
本人的实现很复杂,代码可以参考其他博客,大概方法是把三个串按dfs枚举顺序弄成一个序列,就可以避免8种讨论了
#include <algorithm>
#include <iostream>
#include <cstdlib>
#include <cstring>
#include <cstdio>
#include <cmath>
#define RG register
#define il inline
#define iter iterator
#define Max(a,b) ((a)>(b)?(a):(b))
#define Min(a,b) ((a)<(b)?(a):(b))
using namespace std;
const int N=31;
char S[N];int n,a[N],b[N],c[N],val[N];bool d[N];
il void dfs(int x,int ad){
if(x==0){
if(!ad){
for(int i=1;i<=n;i++)printf("%d ",val[i]);
exit(0);
}
return ;
}
RG int i1=a[x],i2=b[x],i3=c[x],j,i;
if(val[i1]!=-1 && val[i2]!=-1 && val[i3]!=-1){
if((val[i1]+val[i2]+ad)%n==val[i3])
dfs(x-1,val[i1]+val[i2]+ad>=n);
return ;
}
if(val[i1]==-1 && val[i2]==-1 && val[i3]==-1){
if(i1!=i2)
{
for(i=n-1;i>=0;i--){
if(d[i])continue;
d[i]=true;val[i1]=i;
for(j=n-1;j>=0;j--){
if(d[j])continue;
val[i2]=j;val[i3]=(i+j+ad);d[j]=true;
if(val[i3]>=n)val[i3]-=n;
if(!d[val[i3]])
d[val[i3]]=1,dfs(x-1,i+j+ad>=n),d[val[i3]]=0;
else if(i3==i1 || i3==i2)dfs(x-1,i+j+ad>=n);
val[i2]=-1;val[i3]=-1;d[j]=false;
}
d[i]=false;val[i1]=-1;
}
}
else{
for(i=n-1;i>=0;i--){
if(d[i])continue;
d[i]=true;val[i1]=val[i2]=i;
val[i3]=(i+i+ad)%n;
if(!d[val[i3]])
d[val[i3]]=1,dfs(x-1,i+i+ad>=n),d[val[i3]]=0;
else if(i3==i1 || i3==i2)dfs(x-1,i+i+ad>=n);
d[i]=false;val[i1]=val[i2]=-1;
}
}
return ;
}
else if(val[i3]==-1){
if(val[i1]!=-1 && val[i2]!=-1){
val[i3]=(val[i1]+val[i2]+ad)%n;
if(!d[val[i3]])
d[val[i3]]=1,dfs(x-1,val[i1]+val[i2]+ad>=n),d[val[i3]]=0;
else if(i3==i1 || i3==i2)dfs(x-1,val[i1]+val[i2]+ad>=n);
val[i3]=-1;
return ;
}
if(val[i2]==-1)swap(i1,i2);
for(i=n-1;i>=0;i--){
if(d[i])continue;
val[i1]=i;d[i]=true;
val[i3]=(i+val[i2]+ad)%n;
if(!d[val[i3]])
d[val[i3]]=1,dfs(x-1,i+val[i2]+ad>=n),d[val[i3]]=0;
else if(i1==i3 || i3==i2)dfs(x-1,i+val[i2]+ad>=n);
val[i1]=-1;val[i3]=-1;
d[i]=false;
}
return ;
}
if(val[i3]!=-1){
if(val[i1]==-1 && val[i2]==-1){
for(i=n-1;i>=0;i--){
if(d[i])continue;
val[i1]=i;d[i]=true;
if(i+ad<=val[i3])val[i2]=val[i3]-i-ad;
else val[i2]=val[i3]+n-i-ad;
if(!d[val[i2]])
d[val[i2]]=1,dfs(x-1,val[i1]+val[i2]+ad>=n),d[val[i2]]=0;
else if(i2==i1 || i2==i3)dfs(x-1,val[i1]+val[i2]+ad>=n);
val[i1]=-1;val[i2]=-1;
d[i]=false;
}
}
else{
if(val[i2]==-1)swap(i1,i2);
if(val[i2]+ad<=val[i3])val[i1]=val[i3]-val[i2]-ad;
else val[i1]=val[i3]+n-val[i2]-ad;
if(!d[val[i1]])
d[val[i1]]=1,dfs(x-1,val[i1]+val[i2]+ad>=n),d[val[i1]]=0;
else if(i1==i3 || i1==i2)dfs(x-1,val[i1]+val[i2]+ad>=n);
val[i1]=-1;
}
}
}
void work()
{
scanf("%d",&n);
scanf("%s",S+1);memset(val,-1,sizeof(val));
for(int i=1;i<=n;i++)a[i]=S[i]-'A'+1;
scanf("%s",S+1);
for(int i=1;i<=n;i++)b[i]=S[i]-'A'+1;
scanf("%s",S+1);
for(int i=1;i<=n;i++)c[i]=S[i]-'A'+1;
dfs(n,0);
}
int main()
{
work();
return 0;
}
【NOIP2004】虫食算的更多相关文章
- NOIP2004 虫食算
描述 所谓虫食算,就是原先的算式中有一部分被虫子啃掉了,需要我们根据剩下的数字来判定被啃掉的字母.来看一个简单的例子:43#9865#045+ 8468#6633= 44445506678其中#号代表 ...
- [BZOJ1902]:[NOIP2004]虫食算(搜索)
题目传送门 题目描述 所谓虫食算,就是原先的算式中有一部分被虫子啃掉了,需要我们根据剩下的数字来判定被啃掉的字母. 来看一个简单的例子: 43#98650#45+8468#6633=444455069 ...
- Luogu1092 NOIP2004虫食算(搜索+高斯消元)
暴力枚举每一位是否进位,然后就可以高斯消元解出方程了.然而复杂度是O(2nn3),相当不靠谱. 考虑优化.注意到某一位进位情况的变化只会影响到方程的常数项,于是可以在最开始做一次高斯消元算出每个未知数 ...
- [Noip2004]虫食算 dfs
搜索问题的关键:优秀的搜索策略以及行之有效的减枝 对于这道题我们阶乘搜肯定不行所以我们按位搜,我们对每一位的三个数进行赋值,然后判解. 对于此一类的搜索乘上一个几十的常数来减枝往往要比直接搜要快得多, ...
- NOIP 2004 虫食算题解
问题 E: [Noip2004]虫食算 时间限制: 1 Sec 内存限制: 128 MB 题目描述 所谓虫食算,就是原先的算式中有一部分被虫子啃掉了,需要我们根据剩下的数字来判定被啃掉的字母.来看一 ...
- [NOIP2004] 提高组 洛谷P1092 虫食算
题目描述 所谓虫食算,就是原先的算式中有一部分被虫子啃掉了,需要我们根据剩下的数字来判定被啃掉的字母.来看一个简单的例子: 43#9865#045 +8468#6633 44445509678 其中# ...
- 【NOIP2004】【CJOJ1703】【洛谷1092】虫食算
题面 题目描述 所谓虫食算,就是原先的算式中有一部分被虫子啃掉了,需要我们根据剩下的数字来判定被啃掉的字母.来看一个简单的例子: 43#9865#045 +8468#6633 44445509678 ...
- 洛谷 P1092 虫食算 Label:dfs
题目描述 所谓虫食算,就是原先的算式中有一部分被虫子啃掉了,需要我们根据剩下的数字来判定被啃掉的字母.来看一个简单的例子: 43#9865#045 +8468#6633 44445509678 其中# ...
- Luogu P1092 虫食算
题目描述 所谓虫食算,就是原先的算式中有一部分被虫子啃掉了,需要我们根据剩下的数字来判定被啃掉的字母.来看一个简单的例子: 43#9865#045 +8468#6633 44445509678 其中# ...
随机推荐
- 201621123062《java程序设计》第七周作业总结
1. 本周学习总结 1.1 思维导图:Java图形界面总结 1.2 可选:使用常规方法总结其他上课内容. 1.布局管理器的具体使用方法 2.事件处理模型及其代码的编写 3.Swing中的常用组件 4. ...
- 201621123050 《Java程序设计》第7周学习总结
1. 本周学习总结 1.1 思维导图:Java图形界面总结 2.书面作业 1. GUI中的事件处理 1.1 写出事件处理模型中最重要的几个关键词. 1.事件:用户的操作,例如点击或输入之类的操作 2. ...
- 构建微服务开发环境8————Hello 微服务
[内容指引] 1.用IDEA打开微服务项目; 2.更新Maven依赖: 3.IntelliJ IDEA JDK配置; 4.修改代码: 5.运行微服务: 6.将代码变更提交到Github. 经过前面的努 ...
- 创建帧动画1 - xml方式
废话不多说,先看东西 创建帧动画1 - xml方式 帧动画的创建方式主要以下2种: * 用xml创建动画: * 用代码创建动画: 本文内容主要关注 xml文件 创建帧动画的方式 xml文件 ...
- Python基础学习篇章四
一. Python数据类型之字典 1. 键的排序:for循环 由于字典不是序列,因此没有可靠的从左至右的顺序.这就导致当建立一个字典,将它打印出来,它的键也许会以与我们输入时的不同的顺序出现.有时候我 ...
- Django中ORM介绍和字段及其参数
ORM介绍 ORM概念 对象关系映射(Object Relational Mapping,简称ORM)模式是一种为了解决面向对象与关系数据库存在的互不匹配的现象的技术. 简单的说,ORM是通过使用描述 ...
- Java设计模式(八)Proxy代理模式
一.场景描述 代理在生活中并不少见,租房子需要找中介,打官司需要找律师,很多事情我们需要找专业人士代理我们做,另一方面,中介和律师也代理了房东.法律程序与我们打交道. 当然,设计模式中的代理与广义的代 ...
- Oracle处理XML字段时遇到的ORA-31013: XPATH 表达式无效问题
select extractValue(ed.info_id, '/Root/ExpandProfile/PhoneNumber') as phone, extractValue(ed.info_id ...
- ipv4与ipv6的区别
对于计算机网络有一定了解的园园们来说,对这两个概念应该比较熟悉,我也将我知道的一点点小知识分享给大家吧 1.协议使用的广泛程度不同 目前,ipv4得到了广泛的应用,基本上所以的与上网有关的(看电影,玩 ...
- re模块中的compile函数
compile compile(pattern,flag=0) compile a regular expression pattern,return a pattern object compile ...