您的位置:首页 > 其它

USACO-Section3.1 Contact【暴力枚举】

2018-02-12 13:45 337 查看

题目描述:

奶牛们开始对用射电望远镜扫描牧场外的宇宙感兴趣。最近,他们注意到了一种非常奇怪的脉冲调制微波从星系的中央发射出来。他们希望知道电波是否是被某些地外生命发射出来的,还是仅仅是普通的的星星发出的。

帮助奶牛们用一个能够分析他们在文件中记下的记录的工具来找到真相。他们在寻找长度在A到B之间(含)在每天的数据文件中重复得最多的比特序列 (1 <= A <= B <= 12)。他们在找那些重复得最多的比特序列。一个输入限制告诉你应输出多少频率最多的序列。

符合的序列可能会重叠,并且至少重复一次的序列会被计数。

INPUT FORMAT:

第一行: 三个用空格分隔的整数: A, B, N; (1 <= N < 50)

第二行及以后: 一个最多200,000字符的序列,全是0或1; 每行字符数不大于80。

OUTPUT FORMAT:

输出N个频率最高的序列(按照频率由高到低的次序)。由短到长排列频率相同的这些序列,如果长短相同,按二进制大小排列。如果出现的序列个数小于N,输出存在的序列。

对于每个存在的频率,先输出单独包含该频率的一行,再输出以空格分隔的这些频率。每行六个(除非少于六个剩下)。

SIMPLE INPUT

2 4 10

01010010010001000111101100001010011001111000010010011110010000000

在样例里,序列100出现了12次,而序列1000出现了5次。次数最多的序列是00,出现了23次。

SIMPLE OUTPUT

23

00

15

01 10

12

100

11

11 000 001

10

010

8

0100

7

0010 1001

6

111 0000

5

011 110 1000

4

0001 0011 1100

解题思路:

这道题通过枚举就可以解决,主要的问题为如何存储数据,我的方法是在每个字符串前面加个1,就可以转换为数字保存,然后通过stl中的sort函数排序,问题得解。

#include<stdio.h>
#include<string.h>
#include<stdlib.h>
#include<math.h>
#include<algorithm>
using namespace std;
int A,B,N;
int ccount=0;
char a[200010];
FILE *fout;
typedef struct T{//用于保存每种字符串的种类和个数
int x;
int y;
}T;
T ans[10000];
bool cmp(T p,T q){//排序函数
if(p.y>q.y)return true;
else if(p.y==q.y&&p.x<q.x)return true;
else return false;
}
int fun(int t){//将转换成数字的字符串重新反转输出
int tcount=0,temp[13];
while(t){
temp[tcount++]=t&1;
t>>=1;
}
for(int i=tcount-2;i>=0;i--){
fprintf(fout,"%d",temp[i]);
}
}

int main(){
FILE *fin  = fopen ("contact.in", "r");
fout = fopen ("contact.out", "w");
fscanf(fin,"%d%d%d\n",&A,&B,&N);
for(int i=0;i<10000;i++){
ans[i].x=i;
ans[i].y=0;
}
while(fscanf(fin,"%c",&a[ccount])!=EOF){
if(a[ccount]=='\n')continue;
ccount++;
}
for(int i=0;i<ccount;i++){//在每个字符串前面加个1,这样就可以很方便的用数字来记录
long int t=1;
for(int j=0;j<B;j++){
if(i+j>=ccount)break;
else {
t<<=1;
t+=(a[i+j]-'0');
if(j>=A-1){
if(t<0)break;
ans[t].y++;
}
}
}
}
sort(ans,ans+10000,cmp);
int flag=0,anscount;
for(int i=0;flag<=N;i++){
if(ans[i].y>=1){
if(i==0){
fprintf(fout,"%d\n",ans[i].y);
fun(ans[i].x);
anscount=1;
flag++;
}
else if(ans[i-1].y!=ans[i].y){
if(flag==N)break;
fprintf(fout,"\n%d\n",ans[i].y);
fun(ans[i].x);
anscount=1;
flag++;
}
else if(ans[i-1].y==ans[i].y){
if(anscount>0&&anscount%6<=5&&anscount%6>0){
fprintf(fout," ");
fun(ans[i].x);
anscount++;
}
else if(anscount%6==0){
fprintf(fout,"\n");
fun(ans[i].x);
anscount++;
}
}
}
else break;
}
fprintf(fout,"\n");
exit(0);
}
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签:  算法 USACO