4.密文搜索
福尔摩斯从X星收到一份资料,全部是小写字母组成。 他的助手提供了另一份资料:许多长度为8的密码列表。
福尔摩斯发现,这些密码是被打乱后隐藏在先前那份资料中的。
请你编写一个程序,从第一份资料中搜索可能隐藏密码的位置。要考虑密码的所有排列可能性。 数据格式:
输入第一行:一个字符串s,全部由小写字母组成,长度小于1024*1024 紧接着一行是一个整数n,表示以下有n行密码,1<=n<=1000
紧接着是n行字符串,都是小写字母组成,长度都为8 要求输出: 一个整数, 表示每行密码的所有排列在s中匹配次数的总和。例如: 用户输入: aaaabbbbaabbcccc 2 aaaabbbb abcabccc 则程序应该输出: 4
这是因为:第一个密码匹配了3次,第二个密码匹配了1次,一共4次。 资源约定: 峰值内存消耗 < 512M CPU消耗 < 3000ms
请严格按要求输出,不要画蛇添足地打印类似:“请您输入…” 的多余内容。 所有代码放在同一个源文件中,调试通过后,拷贝提交该源码。 注意:
main函数需要返回0 注意: 只使用ANSI C/ANSI C++ 标准,不要调用依赖于编译环境或操作系统的特殊函数。 注意:所有依赖的函数必须明确地在源文件中 #include , 不能通过工程设置而省略常用头文件。 提交时,注意选择所期望的编译器类型。
思路:
对给定的字符串 进行全排列,然后利用find函数查找是否包含在str内部
不知道会不会超时。。。
#include<iostream> #include<string> #include<algorithm> using namespace std; int tol = 0; string str,p; bool check(string a, int k, int i) { if(i > k){ //排除 i ==k 也就是 允许自身交换 for(int j = k; j < i; ++j){ if(a[j] == a[i]) return false; } } return true; } void perm(string a,int k, int m) { if(k == m){ for(int i = 0; i < a.size(); ++i){ printf("%c",a[i]); } cout <<endl; if(str.find(a) != string::npos){ ++tol; } return ; } for(int i = k; i <= m; ++i){ if(check(a,k,i)){ swap(a[i],a[k]); perm(a,k+1,m); swap(a[i],a[k]); } } } int main() { // freopen("C:/Users/zhangwei/Desktop/input.txt","r",stdin); int n,cnt; cin >> str; scanf("%d",&n); int kcase = 0; for(int i = 0 ; i < n; ++i){ cin >> p; perm(p,0,p.size()-1); } cout << tol <<endl; return 0; }
题目因为密码可以全排列,但是不可以中间间隔。故可以统计连续8位字符,对照密码,看各字符出现次数是否一致。
Code:
#include<iostream>
#include<cstring>
using namespace std;
const int N = 1e6+1e5;
int a[N][30];
int b[30];
int main()
{
char s1[N];
char s2[10];
int n;
scanf("%s",s1);
int len = strlen(s1);
for(int i = 0; i <= len-8; ++i){
for(int j = i; j < i+8; ++j){
++a[i][s1[j]-'a'];
}
}
scanf("%d",&n);
int ans = 0;
while(n--){
scanf("%s",s2);
int len2 = strlen(s2);
memset(b,0,sizeof(b));
for(int i = 0; i < len2; ++i){
++b[s2[i]-'a'];
}
for(int i = 0; i <= len-8; ++i){
bool ok = 1;
for(int j = 0; j < 26; ++j){
if(a[i][j] != b[j]){
ok = 0;
break;
}
}
if(ok) ++ans;
}
}
printf("%d\n",ans);
return 0;
}