所谓虫食算,就是原先的算式中有一部分被虫子啃掉了,需要我们根据剩下的数字来判定被啃掉的字母。来看一个简单的例子:
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 个不同的字母分别代表的数字,使得该加法算式成立。输入数据保证有且仅有一组解。
输入的第一行是一个整数 n,代表进制数。
第二到第四行,每行有一个由大写字母组成的字符串,分别代表两个加数以及和。这 33 个字符串左右两端都没有空格,从左到右依次代表从高位到低位,并且恰好有 n 位。
输出一行 n 个用空格隔开的整数,分别代表 A,B,… 代表的数字。
输入
5
ABCED
BDACE
EBBAA
输出
1 0 3 4 2
对于 30% 的数据,保证 n≤10;
对于 50% 的数据,保证 n≤15;
对于 100% 的数据,保证 1≤n≤26。
听说这道题的正解是高斯消元,但我不会,所以这里提供一种深搜+剪枝的方法。
难点1:n进制加法。接触可能较少,一般接触的都是十进制,所以说一些加减过程可能会比十进制稍困难,不过也挺简单,把十进制的满十进一改为满n进一,总的来说就是把所有十换成n即可。
难点2:怎么剪枝?通过分析,可以分析出:1.如果后面的没进位的话那么会有:(加数1的第k位 + 加数二的第k位) mod n = 和的第k位。2.如果后面的进位了也会有:加数(加数1的第k位 + 加数二的第k位 + 1) mod n = 和的第k位。所以我们只需判断如果有一位k,加数(加数1的第k位 + 加数二的第k位) mod n 与加数(加数1的第k位 + 加数二的第k位 + 1) mod n 都不等于和的第k位,那么这样的情况是不合法的,只需剪掉这样的情况即可。
这里给出剪枝代码:
bool pd() {
for(int i = 1;i <= n;i ++) {
int u = ans[a[i]],v = ans[b[i]],z = ans[c[i]];
//ans[a[i]],ans[b[i]],ans[c[i]]分别表示加数一、加数二与和的第i位
if(u == -1 || v == -1 || z == -1)//如果这个地方还没有被赋过值,跳过
continue;
if((u + v) % n != z && (u + v + 1) % n != z)//如果不合法
return false;
}
return true;
}
难点3:怎么判断答案合不合法?这一个步骤不难,当已将搜索完了n个字母后,将每一个加数与和求出,判断是否相等即可。
给出代码,略过:
bool check() {
int ans1 = 0,ans2 = 0,ans3 = 0;
for(int i = 1;i <= n;i ++) {
ans1 = ans1 * n + ans[a[i]];
ans2 = ans2 * n + ans[b[i]];
ans3 = ans3 * n + ans[c[i]];
}
return ans1 + ans2 == ans3;
}
简单理一下,现将字符转化成数字保存下来,再用一个数组存储,以方便深搜,与处理好后开始深搜,按照字母个数深搜,在中间设一个剪枝,减去不合适的,让后枚举字符,判断是否出现过,如果没出现过便向下搜,直到搜到有答案,并且答案正确为止。
#include
#include
#include
using namespace std;
const int MAXN = 30;
int n,num;
int a[MAXN],b[MAXN],c[MAXN],vis[MAXN],ne[MAXN],ans[MAXN];
char s1[MAXN],s2[MAXN],s3[MAXN];
void Get(int x) {//存储
if(!vis[x]) {
vis[x] = 1;
ne[++ num] = x;
}
}
bool check() {//判断答案
int ans1 = 0,ans2 = 0,ans3 = 0;
for(int i = 1;i <= n;i ++) {
ans1 = ans1 * n + ans[a[i]];
ans2 = ans2 * n + ans[b[i]];
ans3 = ans3 * n + ans[c[i]];
}
return ans1 + ans2 == ans3;
}
bool pd() {//剪枝
for(int i = 1;i <= n;i ++) {
int u = ans[a[i]],v = ans[b[i]],z = ans[c[i]];
if(u == -1 || v == -1 || z == -1)
continue;
if((u + v) % n != z && (u + v + 1) % n != z)
return false;
}
return true;
}
void dfs(int x) {//搜索
if(x == n + 1) {
if(check()) {
for(int i = 0;i < n;i ++)
printf("%d ",ans[i]);//输出答案
exit(0);
}
return;
}
if(!pd())
return;
for(int i = n - 1;i >= 0;i --) {
if(!vis[i]) {//没出现过
vis[i] = 1;
ans[ne[x]] = i;
dfs(x + 1);
ans[ne[x]] = -1;
vis[i] = 0;
}
}
return;
}
int main() {
scanf("%d",&n);
scanf("%s%s%s",s1 + 1,s2 + 1,s3 + 1);
for(int i = n;i >= 1;i --) {
a[i] = s1[i] - 'A';//字符转数字
b[i] = s2[i] - 'A';
c[i] = s3[i] - 'A';
}
for(int i = n;i >= 1;i --) {
Get(a[i]);//存储
Get(b[i]);
Get(c[i]);
}
memset(ans,-1,sizeof(ans));
for(int i = n;i >= 0;i --)
vis[i] = 0;
dfs(1);//开搜
return 0;
}