codevs1064 虫食算
所谓虫食算,就是原先的算式中有一部分被虫子啃掉了,需要我们根据剩下的数字来判定被啃掉的字母。来看一个简单的例子:
43#9865#045
+ 8468#6633
44445506978
其中#号代表被虫子啃掉的数字。根据算式,我们很容易判断:第一行的两个数字分别是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个不同的字母分别代表的数字,使得该加法算式成立。输入数据保证有且仅有一组解,
输入描述
Input Description
输入包含4行。第一行有一个正整数N(N<=26),后面的3行每行有一个由大写字母组成的字符串,分别代表两个加数以及和。这3个字符串左右两端都没有空格,从高位到低位,并且恰好有N位。
输出描述
Output Description
输出包含一行。在这一行中,应当包含唯一的那组解。解是这样表示的:输出N个数字,分别表示A,B,C……所代表的数字,相邻的两个数字用一个空格隔开,不能有多余的空格。
样例输入
Sample Input
5
ABCED
BDACE
EBBAA
样例输出
Sample Output
1 0 3 4 2
数据范围及提示
Data Size & Hint
对于30%的数据,保证有N<=10;
对于50%的数据,保证有N<=15;
对于全部的数据,保证有N<=26。
仔细查了一下错,发现又犯低级错误了。首先我的下标从0开始那么我的搜索边界应该是-1,然而我只到了0,导致有个解没搜到。然后我打的只是裸搜索,没有一个很优秀的剪枝,就是每一步都check一下,还没搜到的等式是不是已经三个都出来了,如果已经都确定了,但是不管进不进位都无法做到的话,那就说明不合法,可以直接return。这样剪枝的话就跑得很快了,基本0.008s就可以过了。
其余的就是裸搜索啦:
//It is made by jump~
#include <iostream>
#include <cstdlib>
#include <cstring>
#include <cstdio>
#include <cmath>
#include <algorithm>
#include <ctime>
#include <vector>
#include <queue>
#include <map>
#include <set>
using namespace std;
typedef long long LL;
#define RG register
const int MAXN = ;
int n,ce,a[MAXN],b[MAXN],c[MAXN];
int match[MAXN],ans[MAXN];
char ch[MAXN];
bool ok,vis[MAXN]; inline int getint()
{
RG int w=,q=; RG char c=getchar();
while((c<'' || c>'') && c!='-') c=getchar(); if(c=='-') q=,c=getchar();
while (c>='' && c<='') w=w*+c-'', c=getchar(); return q ? -w : w;
} inline void dfs(RG int x,RG int type,RG int remain){
if(ok) return ; RG int ans1,ans2,ans3;
for(int i=;i<x;i++) {//进位最多是ce,如果某一位已经确定,但是进位不够则不合法
ans1=match[a[i]],ans2=match[b[i]],ans3=match[c[i]];
if(ans1==-||ans2==-||ans3==-) continue;
if( ((ans3-ans2-ans1)%n+n)%n >ce) return ;
} if(x==-) {//!!!!!!!
for(RG int i=;i<n;i++) { ans[i]=match[i]; if(match[i]==-) return ; }
ok=true; return ;
}
if(type==) {//搜第一个
if(match[a[x]]!=-) dfs(x,type+,remain);
else{
for(RG int i=n-;i>=;i--) {
if(vis[i]) continue;
vis[i]=; match[a[x]]=i; dfs(x,type+,remain);
vis[i]=; match[a[x]]=-;
}
}
}
else{//搜第二个加数的这一位
ans1=match[a[x]],ans2=match[b[x]],ans3=match[c[x]];
if(ans2!=-) {
RG int sheng=ans1+ans2+remain;
if(sheng%n==ans3) dfs(x-,,sheng/n);
else if(ans3==-) {
vis[sheng%n]=; match[c[x]]=sheng%n; dfs(x-,,sheng/n);
vis[sheng%n]=; match[c[x]]=-;
}
}
else{
RG int cc,ss;
for(RG int i=n-;i>=;i--) {
if(vis[i]) continue;
ss=ans1+i+remain; cc=ss%n;
if(cc!=ans3 && ans3!=-) continue;
if(ans3==-) {
if(vis[cc]) continue;
vis[cc]=; match[c[x]]=cc;
}
vis[i]=; match[b[x]]=i;
dfs(x-,,ss/n);
vis[i]=; match[b[x]]=-;
if(ans3==-) {
vis[cc]=; match[c[x]]=-;
}
}
}
}
} inline void work(){
n=getint(); for(RG int i=;i<=n;i++) match[i]=-; ce=*n-; ce/=n;
scanf("%s",ch); for(RG int i=;i<n;i++) a[i]=ch[i]-'A';
scanf("%s",ch); for(RG int i=;i<n;i++) b[i]=ch[i]-'A';
scanf("%s",ch); for(RG int i=;i<n;i++) c[i]=ch[i]-'A';
ok=false; dfs(n-,,);
printf("%d",ans[]); for(RG int i=;i<n;i++) printf(" %d",ans[i]);
} int main()
{
work();
return ;
}
codevs1064 虫食算的更多相关文章
- 洛谷 P1092 虫食算 Label:dfs
题目描述 所谓虫食算,就是原先的算式中有一部分被虫子啃掉了,需要我们根据剩下的数字来判定被啃掉的字母.来看一个简单的例子: 43#9865#045 +8468#6633 44445509678 其中# ...
- 深度优先搜索 codevs 1064 虫食算
codevs 1064 虫食算 2004年NOIP全国联赛提高组 时间限制: 2 s 空间限制: 128000 KB 题目等级 : 钻石 Diamond 题目描述 Description 所 ...
- NOIP2004 虫食算
描述 所谓虫食算,就是原先的算式中有一部分被虫子啃掉了,需要我们根据剩下的数字来判定被啃掉的字母.来看一个简单的例子:43#9865#045+ 8468#6633= 44445506678其中#号代表 ...
- Codevs 1064 虫食算 2004年NOIP全国联赛提高组
1064 虫食算 2004年NOIP全国联赛提高组 时间限制: 2 s 空间限制: 128000 KB 题目等级 : 钻石 Diamond 题目描述 Description 所谓虫食算,就是原先的算式 ...
- Luogu P1092 虫食算
题目描述 所谓虫食算,就是原先的算式中有一部分被虫子啃掉了,需要我们根据剩下的数字来判定被啃掉的字母.来看一个简单的例子: 43#9865#045 +8468#6633 44445509678 其中# ...
- 【NOIP2004】【CJOJ1703】【洛谷1092】虫食算
题面 题目描述 所谓虫食算,就是原先的算式中有一部分被虫子啃掉了,需要我们根据剩下的数字来判定被啃掉的字母.来看一个简单的例子: 43#9865#045 +8468#6633 44445509678 ...
- 【NOIP2004】虫食算
Description 所谓虫食算,就是原先的算式中有一部分被虫子啃掉了,需要我们根据剩下的数字来判定被啃掉的字母.来看一个简单的例子: 43#9865#045 +. 8468#6633 444455 ...
- 洛谷P1092 虫食算
P1092 虫食算 题目描述 所谓虫食算,就是原先的算式中有一部分被虫子啃掉了,需要我们根据剩下的数字来判定被啃掉的字母.来看一个简单的例子: http://paste.ubuntu.com/2544 ...
- 【题解】 P1092虫食算
[题解]P1092 虫食算 老题了,很经典. 用到了一些搜索套路. 可行性剪枝,劣者靠后,随机化,\(etc......\) 搜索设参也很有技巧,设一个\(adjustment\)参数可以很方便地在两 ...
随机推荐
- Unity3D手势及重力加速度(神庙逃亡操作)
Unity实现神庙逃亡操作 现在特别火的跑酷游戏<神庙逃亡>是用Unity3D引擎开发的 游戏的操作:用手指拨动(划动)人物就转向,利用手机的重力感应进行人物左右调整. 今天用Unity来 ...
- eclipse的使用-------Text File Encoding没有GBK选项的设置
eclipse的使用-------Text File Encoding没有GBK选项的设置 2013-12-25 09:48:06 标签:java myeclipse使用 有一个项目是使用GBK编码的 ...
- templatecolumn checkcolumn
- OXM
O/X Mapper 是什么? Spring 3.0 的一个新特性是 O/X Mapper.O/X 映射器这个概念并不新鲜,O 代表 Object,X 代表 XML.它的目的是在 Java 对象(几乎 ...
- Java中的IO流系统详解(转载)
摘要: Java 流在处理上分为字符流和字节流.字符流处理的单元为 2 个字节的 Unicode 字符,分别操作字符.字符数组或字符串,而字节流处理单元为 1 个字节,操作字节和字节数组. Java ...
- wireshark排查打印机问题
抓包工具排除故障 前言:上网慢,可能是内网堵了.装上wireshark,可抓到广播包,多播包,以及发给自己的包.如果想抓lan内其他人之间的通信包,那就要在sw上做端口镜像. 背景 调试打印机的人发现 ...
- dos常用命令
进入终端 首先具备一个控制台(命令行提示符窗口)用于输入dos命令: 打开一个控制台的方式: 方式一:开始-------> 所有程序--------->附件----------->命 ...
- Linq To Entities 及其相关(进阶)
上篇我们讲解了Linq To Entities的一些基本操作,这篇我们主要是讲解一些比较高级的东西:存储过程查询,SQL语句查询以及表达式树. 存储过程 首先来讲解存储过程查询. //Query a ...
- JSON与JSONP
JSON JSON:一种用于在浏览器和服务器之间交换信息的基于文本的轻量级数据格式.是JS对象的字符串表示.例如:‘{''name":"aa","age&quo ...
- caffe windows 学习第一步:编译和安装(vs2012+win 64)
没有GPU,没有linux, 只好装caffe的windows版本了. 我的系统是win10(64位),vs 2012版本,其它什么都没有装,因此会需要一切的依赖库. 其实操作系统只要是64位就行了, ...