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

题目描述:

奶牛们开始对用射电望远镜扫描牧场外的宇宙感兴趣。最近,他们注意到了一种非常奇怪的脉冲调制微波从星系的中央发射出来。他们希望知道电波是否是被某些地外生命发射出来的,还是仅仅是普通的的星星发出的。
帮助奶牛们用一个能够分析他们在文件中记下的记录的工具来找到真相。他们在寻找长度在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);
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值