[luogu 1092] 虫食算 (暴力搜索剪枝)
Description
Input
包含四行。
第一行有一个正整数 (N≤26)。
后面的三行,每行有一个由大写字母组成的字符串,分别代表两个加数以及和。这3个字符串左右两端都没有空格,从高位到低位,并且恰好有N位。
Output
一行,即唯一的那组解。
解是这样表示的:输出NN个数字,分别表示A,B,C,…所代表的数字,相邻的两个数字用一个空格隔开,不能有多余的空格。
Sample Input
5
ABCED
BDACE
EBBAA
Sample Output
1 0 3 4 2
INIT
对于30%的数据,保证有N≤10;
对于50%的数据,保证有N≤15;
对于全部的数据,保证有N≤26。
noip2004提高组第4题
Solution
暴力搜索剪枝没啥好说的。。。
写完才明白按照一列搜索比按字母搜索要快好多,也好剪枝
于是我写了一大堆特判才卡过
Code
丑陋的代码
#include <cmath>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <iostream>
#include <algorithm>
#define F(i,a,b) for(register int i=(a);i<=(b);i++)
#define R(i,a,b) for(register int i=(b);i>=(a);i--)
using namespace std;
inline int read() {
int x=0,f=1;char c=getchar();
while(!isdigit(c)) {if(c=='-')f=-f;c=getchar();}
while(isdigit(c)) x=(x<<1)+(x<<3)+c-48,c=getchar();
return x*f;
}
const int N=30;
bool flag;
int n,tot,cnt;
bool vis[N];
int A[N],B[N],C[N],mp[N],sta[N],deg[N],id[N];
char ch[N];
bool ck() {
int A1=mp[A[1]],B1=mp[B[1]],C1=mp[C[1]];
if(~A1&&~C1&&A1>C1) return 0;
if(~B1&&~C1&&B1>C1) return 0;
if(~A1&&~B1) {
if(A1+B1>tot-1) return 0;
if(~C1&&A1+B1!=C1&&A1+B1+1!=C1) return 0;
}
int las=0;
R(i,1,n) {
int CA=A[i],CB=B[i],CC=C[i];
int &DA=mp[A[i]],&DB=mp[B[i]],&DC=mp[C[i]];
if(~DA&&~DB&&~DC) {
if(las==-1&&(DA+DB+1)%tot!=DC&&(DA+DB)%tot!=DC) return 0;
if(~las&&(DA+DB+las)%tot!=DC) return 0;
if(~las) las=(DA+DB+las)/n;
if(las==-1&&DA+DB>=n) las=1;
}
else if(~DA&&~DB) {
if(~las&&vis[(DA+DB+las)%tot]) return 0;
if(las==-1&&vis[(DA+DB)%tot]&&vis[(DA+DB+1)%tot]) return 0;
if(~las&&DC==-1) DC=(DA+DB+las)%tot,vis[DC]=1,sta[++cnt]=CC;
}
else if(~DA&&~DC) {
int dc=DC; if(dc<=DA) dc+=tot;
if(~las&&vis[dc-DA-las]) return 0;
if(las==-1&&vis[dc-DA]&&vis[dc-DA-1]) return 0;
if(~las&&DB==-1) DB=dc-DA-las,vis[DB]=1,sta[++cnt]=CB;
}
else if(~DB&&~DC) {
int dc=DC; if(dc<=DB) dc+=tot;
if(~las&&vis[dc-DB-las]) return 0;
if(las==-1&&vis[dc-DB]&&vis[dc-DB-1]) return 0;
if(~las&&DA==-1) DA=dc-DB-las,vis[DA]=1,sta[++cnt]=CA;
}
if(~DA&&~DB&&~las) las=(DA+DB+las)/n;
if(las==-1&&~DA&&~DB&&DA+DB>=n) las=1;
if(DA==-1||DB==-1||DC==-1) las=-1;
}
return 1;
}
void dfs(int pos) {
if(flag) return ;
if(~mp[id[pos]]) {dfs(pos+1);return ;}
if(pos==tot+1) {
F(i,1,tot) printf("%d ",mp[i]);
flag=1; return ;
}
F(i,0,tot-1) if(!vis[i]) {
mp[id[pos]]=i; vis[i]=1; int ls=cnt;
if(ck()) dfs(pos+1);
mp[id[pos]]=-1; vis[i]=0;
while(cnt>ls) vis[mp[sta[cnt]]]=0,mp[sta[cnt]]=-1,cnt--;
if(flag) return ;
}
}
bool cmp(int x,int y) {return deg[x]>deg[y];}
int main() {
n=read();
scanf("%s",ch+1);F(i,1,n) {A[i]=ch[i]-'A'+1;if(!vis[A[i]]) tot++,vis[A[i]]=1;deg[A[i]]++;}
scanf("%s",ch+1);F(i,1,n) {B[i]=ch[i]-'A'+1;if(!vis[B[i]]) tot++,vis[B[i]]=1;deg[A[i]]++;}
scanf("%s",ch+1);F(i,1,n) {C[i]=ch[i]-'A'+1;if(!vis[C[i]]) tot++,vis[C[i]]=1;deg[A[i]]++;}
memset(mp,-1,sizeof(mp)); memset(vis,0,sizeof(vis));
F(i,1,n) id[i]=i;
sort(id+1,id+1+n,cmp);
F(i,1,n) if(A[i]==B[i]&&B[i]==C[i]) {mp[A[i]]=0;vis[0]=1;dfs(1);return 0;}
F(i,1,n) if((A[i]==C[i])&&A[i]!=B[i]) {
mp[B[i]]=0;vis[0]=1;dfs(1);mp[B[i]]=-1;vis[0]=0;
if(!flag) {mp[A[i]]=0;vis[0]=1;mp[B[i]]=tot-1;vis[tot-1]=1;dfs(1);}
return 0;
}
F(i,1,n) if((B[i]==C[i])&&A[i]!=B[i]) {
mp[A[i]]=0;vis[0]=1;dfs(1);mp[A[i]]=-1;vis[0]=0;
if(!flag) {mp[B[i]]=0;vis[0]=1;mp[A[i]]=tot-1;vis[tot-1]=1;dfs(1);}
return 0;
}
dfs(1);
return 0;
}
[luogu 1092] 虫食算 (暴力搜索剪枝)的更多相关文章
- Luogu P1092 虫食算【搜索/剪枝】 By cellur925
题目传送门 这道题是一道经久不衰的搜索题目,但是开始做的时候我没什么思路==.初始值-1 输出格式 \(naive\)想法 从右往左依次尝试填充数字,把算式当做一个3行\(n\)列的网格.(什么?你问 ...
- Luogu P1092 虫食算(枚举+剪枝)
P1092 虫食算 题面 题目描述 所谓虫食算,就是原先的算式中有一部分被虫子啃掉了,需要我们根据剩下的数字来判定被啃掉的字母.来看一个简单的例子: 43#9865#045 + 8468#6633 4 ...
- [BZOJ1902]:[NOIP2004]虫食算(搜索)
题目传送门 题目描述 所谓虫食算,就是原先的算式中有一部分被虫子啃掉了,需要我们根据剩下的数字来判定被啃掉的字母. 来看一个简单的例子: 43#98650#45+8468#6633=444455069 ...
- Luogu P1092 虫食算
题目描述 所谓虫食算,就是原先的算式中有一部分被虫子啃掉了,需要我们根据剩下的数字来判定被啃掉的字母.来看一个简单的例子: 43#9865#045 +8468#6633 44445509678 其中# ...
- Luogu1092 NOIP2004虫食算(搜索+高斯消元)
暴力枚举每一位是否进位,然后就可以高斯消元解出方程了.然而复杂度是O(2nn3),相当不靠谱. 考虑优化.注意到某一位进位情况的变化只会影响到方程的常数项,于是可以在最开始做一次高斯消元算出每个未知数 ...
- Luogu P1092 虫食算 爆搜
心路历程:震惊,我竟然得了$90$分!!...康康数据...奥..(忽然有了邪恶的卡数据的想法) 于是把$for(int \space i=0;i<n;++i)$改成了$for(int \spa ...
- 【NOIP2004】【CJOJ1703】【洛谷1092】虫食算
题面 题目描述 所谓虫食算,就是原先的算式中有一部分被虫子啃掉了,需要我们根据剩下的数字来判定被啃掉的字母.来看一个简单的例子: 43#9865#045 +8468#6633 44445509678 ...
- 【搜索】$P1092$虫食算
题目链接 首先,我们只考虑加法的虫食算.这里的加法是N进制加法,算式中三个数都有N位,允许有前导的0. 其次,虫子把所有的数都啃光了,我们只知道哪些数字是相同的,我们将相同的数字用相同的字母表示,不同 ...
- P1092 虫食算 题解(搜索)
题目链接 P1092 虫食算 解题思路 好题啊!这个搜索好难写...... 大概是要考虑进位和考虑使用过某个数字这两个东西,但就很容易出错...... 首先这个从后往前搜比较好想,按照从后往前出现的顺 ...
随机推荐
- java.text.ParseException: Unparseable date: "2015-06-09 hh:56:19"
1.错误描写叙述 [DEBUG:]2015-06-09 16:56:19,520 [-------------------transcation start!--------------] java. ...
- Codeforces Round #276 (Div. 1)B. Maximum Value 筛法
D. Maximum Value You are given a sequence a consisting of n integers. Find the maximum possible ...
- U4704 函数
U4704 函数 题目背景 设gcd(a,b)为a和b的最大公约数,xor(a,b)为a异或b的结果. 最大公约数 异或 题目描述 kkk总是把gcd写成xor.今天数学考试恰好出到了gcd(a,b) ...
- FreeBSD内核之中的一个 ALQ机制的使用
背景: 笔者由于一个项目,这段时间在使用FreeBSD进行内核模块的编程. 之前做过一段时间的Linux下驱动模块编程.对Linux下的模块编程还算熟悉. 如今突然转到FreeBSD下.尽管Linux ...
- Too-Java:Intellij Idea
ylbtech-Too-Java:Intellij Idea IDEA 全称 IntelliJ IDEA,是java编程语言开发的集成环境.IntelliJ在业界被公认为最好的java开发工具之一,尤 ...
- 第2章 安装Nodejs 2-2 Nodejs版本常识
- 机器学习——Day 2 简单线性回归
写在开头 由于某些原因开始了机器学习,为了更好的理解和深入的思考(记录)所以开始写博客. 学习教程来源于github的Avik-Jain的100-Days-Of-MLCode 英文版:https:// ...
- Cordova 开发环境搭建及创建第一个app
整理记录使用cordova创建app应用程序并将其部署至Android系统移动设备上操作过程,具体如下: 一.前期安装环境 1. 安装JDK(java开发工具包) 2. 安装gradle 3. 安装A ...
- Phoenix与Squirrel 是什么?
不多说,直接上干货! 前言 Phoenix是HBase的开源SQL引擎. squirrel是windows上Phoneix可视化工具. Phoenix的官网 http://phoenix.apach ...
- springboot 的一般配置
import javax.servlet.Filter; import org.springframework.boot.SpringApplication; import org.springfra ...