2017-9-3模拟赛T3 密码(key)

时间:2023-03-08 19:23:14

题目


2017-9-3模拟赛T3 密码(key)

题解


这题用类似暴力+优化(划掉)的思想。

对于每个轨迹串,求出每一位向后的第一个0-9间某个数字的位置(如123112中3后面第1个2的位置为从左往右数第6个),复杂度O(Σn)=O(L)。

在dfs中枚举密码,相邻两位相同的在处理中合并(即不枚举1223,改为123)。

还有就是要注意,当位数为2或3时,若匹配成功则答案+3而不是+1。比如枚举23,可以有2333,2233,2223三种密码。枚举123,可以有1123,1223,1233三种。

最后就是细节问题了。

代码


 #include <stdio.h>
#define N 1005
int n,a[N],d[],k[]={,,,,},ans=;char *s[N];
struct sufarr {
short a[];
void clear(){for(int i=;i<;++i) a[i]=;}
} *suf[N],tmp;
inline void input() {
scanf("%d",&n);
for(int i=;i<n;++i) {
scanf("%d",a+i);
s[i]=new char[a[i]+];
suf[i]=new sufarr[a[i]+];
scanf("%s",s[i]+);
tmp.clear();
for(int j=a[i];j;--j) {
suf[i][j]=tmp;
tmp.a[s[i][j]^]=j;
}
suf[i][]=tmp;
}
}
inline bool test(int len) {
int cnt,cur,u;
for(int i=;i<n;++i) {
for(cur=,cnt=;cnt<=len;++cnt) {
u=suf[i][cur].a[d[cnt]];
if(!u) break;cur=u;
}
if(cnt<=len) return ;
}
return ;
}
void dfs(int t) {
for(int i=;i<;++i) {
if(i!=d[t-]) {
d[t]=i;
if(test(t)) {
ans+=k[t];
if(t<) dfs(t+);
}
}
}
}
int main() {
input();
d[]=-;dfs();
printf("%d",ans);
return ;
}

#include <OI必会>

using namespace OIer;

void 完结撒花() {

日常 %("czhou");

}

int main() {

完结撒花();

return 0;

}