P1092 虫食算——题解

题目传送

P1092 虫食算——题解

P1092 虫食算——题解

P1092 虫食算——题解

(据说官方正解为高斯消元,但用搜索也能过,这里就讲讲搜索算法吧。)

  对于一道搜索题,首先考虑一下大体怎样搜索。因为要考虑加法的进位,所以从左往右搜索对于考虑进位来说十分麻烦,而从右往左搜索就没有这种麻烦,故搜索顺序从右往左。但是发现整个式子的一位上由三个字符串的一位组成,且这三个分别担当加数、结果中的一部分,逐个搜索的话还要麻烦地分类讨论,考虑再优化一下搜索顺序。发现一共只有n个不同的数和字母,并且每个数和字母都至少出现一次,那么可以从右往左找出字母第一次出现的位置并存到next数组里,只要按照next数组搜一遍就完事了。

  显然一个裸搜在此题肯定会TLE了,考虑一下剪枝。由于每个字符串都有n位,那么作为加数的两个字符串的最高位加进位加起来一定不能大于等于n导致进位。同时如果发现在同一位的数若都已经确定,发现两加数那一位上的数的和mod n(不进位的情况)不等于结果那一位上的数,并且两加数那一位上的数的和+1再mod n(进位的情况)仍然不等于结果那一位上的数,显然是不合题意的,故要剪掉。

  一个小技巧:发现相同的字母代表相同的数。我们可以把字母数字化,即每个字母都减'A'加1,这样若原字母为'A',处理后就为1,若为'B',就为2,...以此类推。这样可以让处理后的值作为最后记录答案的数组num的下标,同一个字母处理后的变为的下标也一样,故可以统一处理。

见AC代码:

 #include<iostream>
#include<cstdio>
#include<cstring>
#include<cstdlib> using namespace std; int n; char s1[],s2[],s3[]; int a[],b[],c[],next[],cnt,num[]; bool used[]; inline void getused(int x)//处理出next数组
{
if(!used[x])
{
used[x]=;
next[++cnt]=x;
}
} inline int canprune()//剪枝判断 prune:剪枝(来自翻译)
{
if(num[a[]]+num[b[]]>=n)
return ;
int A,B,C;
for(int i=n;i>=;i--)
{
A=num[a[i]],B=num[b[i]],C=num[c[i]];
if(A==-||B==-||C==-) continue;
if((A+B)%n!=C&&(A+B+)%n!=C) return ;
}
return ;
} inline int judge()//结果判断
{
if(num[a[]]+num[b[]]>=n)
return ;
int A,B,C,x=;
for(int i=n;i>=;i--)
{
A=num[a[i]],B=num[b[i]],C=num[c[i]];
if((A+B+x)%n!=C) return ;
x=(A+B+x)/n;
}
return ;
} void print()
{
for(int i=;i<n;i++)
printf("%d ",num[i]);
cout<<num[n];
exit();//要用 stdlib.h 或 cstdlib库(少了头文件的话本地虽然能过,但交上去就回CE),作用是直接退出程序。
} void dfs(int k)
{
for(int i=n-;i>=;--i)
if(!used[i])
{
num[next[k]]=i;
used[i]=;
if(k==n)
{
if(judge())
print();
}
else
{
if(!canprune())
dfs(k+);
}
num[next[k]]=-;
used[i]=;
}
} int main()
{
scanf("%d",&n);
scanf("%s%s%s",s1,s2,s3);
for(int i=n;i>=;--i)
{
a[i]=s1[i-]-'A'+;
getused(a[i]);
b[i]=s2[i-]-'A'+;
getused(b[i]);
c[i]=s3[i-]-'A'+;
getused(c[i]);
}
memset(num,-,sizeof num);
memset(used,,sizeof used);
dfs();
return ;
}
上一篇:Memcached Client的释疑


下一篇:shell之使用paste命令按列拼接多个文件