zoukankan      html  css  js  c++  java
  • BZOJ2251: [2010Beijing Wc]外星联络

    2251: [2010Beijing Wc]外星联络

    Time Limit: 30 Sec  Memory Limit: 256 MB
    Submit: 365  Solved: 191
    [Submit][Status]

    Description

    小 P 在看过电影《超时空接触》(Contact)之后被深深的打动,决心致力于寻
    找外星人的事业。于是,他每天晚上都爬在屋顶上试图用自己的收音机收听外星
    人发来的信息。虽然他收听到的仅仅是一些噪声,但是他还是按照这些噪声的高
    低电平将接收到的信号改写为由 0 和 1 构成的串, 并坚信外星人的信息就隐藏在
    其中。他认为,外星人发来的信息一定会在他接受到的 01 串中重复出现,所以
    他希望找到他接受到的 01 串中所有重复出现次数大于 1 的子串。但是他收到的
    信号串实在是太长了,于是,他希望你能编一个程序来帮助他。

    Input

    输入文件的第一行是一个整数N ,代表小 P 接收到的信号串的长度。
    输入文件第二行包含一个长度为N 的 01 串,代表小 P 接收到的信号串。

    Output

    输出文件的每一行包含一个出现次数大于1 的子串所出现的次数。输出的顺
    序按对应的子串的字典序排列。

    Sample Input

    7
    1010101

    Sample Output

    3
    3
    2
    2
    4
    3
    3
    2
    2

    HINT

      对于 100%的数据,满足 0 <=  N     <=3000 

    Source

    题解:

    终于1A了一道题。。。

    考虑到以sa[i]开头的本质不同的字串的数目只有 n-sa[i]-h[i]个,而如果按字典序来就从sa[1]-sa[n]即可

    然后求一个字串出现的次数,只要在该字串第一次出现的时候 暴力像后匹配即可,

    我想到了一个优化,从长的向短匹配,这样长的能匹配到的地方短的也能匹配到,所以能充分利用原来计算过的信息。

    可是还是没跑进第一页T_T

    代码:

      1 #include<cstdio>
      2 
      3 #include<cstdlib>
      4 
      5 #include<cmath>
      6 
      7 #include<cstring>
      8 
      9 #include<algorithm>
     10 
     11 #include<iostream>
     12 
     13 #include<vector>
     14 
     15 #include<map>
     16 
     17 #include<set>
     18 
     19 #include<queue>
     20 
     21 #include<string>
     22 
     23 #define inf 1000000000
     24 
     25 #define maxn 5000+100
     26 
     27 #define maxm 500+100
     28 
     29 #define eps 1e-10
     30 
     31 #define ll long long
     32 
     33 #define pa pair<int,int>
     34 
     35 #define for0(i,n) for(int i=0;i<=(n);i++)
     36 
     37 #define for1(i,n) for(int i=1;i<=(n);i++)
     38 
     39 #define for2(i,x,y) for(int i=(x);i<=(y);i++)
     40 
     41 #define for3(i,x,y) for(int i=(x);i>=(y);i--)
     42 
     43 #define mod 1000000007
     44 
     45 using namespace std;
     46 
     47 inline int read()
     48 
     49 {
     50 
     51     int x=0,f=1;char ch=getchar();
     52 
     53     while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();}
     54 
     55     while(ch>='0'&&ch<='9'){x=10*x+ch-'0';ch=getchar();}
     56 
     57     return x*f;
     58 
     59 }
     60 int a[maxn],b[maxn],sa[maxn],wa[maxn],wb[maxn],wr[maxn],h[maxn],rank[maxn];
     61 int n,m,tot;
     62 inline bool cmp(int *r,int a,int b,int l)
     63 {
     64     return r[a]==r[b]&&r[a+l]==r[b+l];
     65 }
     66 void da(int n,int m)
     67 {
     68     int *x=wa,*y=wb,*t;
     69     for0(i,m)wr[i]=0;
     70     for0(i,n)wr[x[i]=a[i]]++;
     71     for1(i,m)wr[i]+=wr[i-1];
     72     for3(i,n,0)sa[--wr[x[i]]]=i;
     73     for(int j=1,p=1;p<=n;j<<=1,m=p)
     74     {
     75         p=0;int i;
     76         for2(i,n-j+1,n)y[p++]=i;
     77         for0(i,n)if(sa[i]>=j)y[p++]=sa[i]-j;
     78         for0(i,m)wr[i]=0;
     79         for0(i,n)wr[x[y[i]]]++;
     80         for1(i,m)wr[i]+=wr[i-1];
     81         for3(i,n,0)sa[--wr[x[y[i]]]]=y[i];
     82         for(i=1,t=x,x=y,y=t,p=1,x[sa[0]]=0;i<=n;i++)
     83             x[sa[i]]=cmp(y,sa[i-1],sa[i],j)?p-1:p++;
     84     }
     85 }
     86 void geth(int n)
     87 {
     88     for1(i,n)rank[sa[i]]=i;
     89     for(int i=0,j,k=0;i<n;h[rank[i++]]=k)
     90      for(k?k--:0,j=sa[rank[i]-1];a[i+k]==a[j+k];k++);
     91 } 
     92 
     93 int main()
     94 
     95 {
     96 
     97     freopen("input.txt","r",stdin);
     98 
     99     freopen("output.txt","w",stdout);
    100 
    101     n=read();
    102     for1(i,n)
    103     {
    104         char ch=' ';
    105         while(ch!='0'&&ch!='1')ch=getchar();
    106         a[i-1]=ch-'0'+1;
    107     }
    108     a[n]=0;
    109     da(n,3);
    110     geth(n);
    111     //for0(i,n)cout<<i<<' '<<sa[i]<<' '<<h[i]<<endl;
    112     for1(i,n-1)
    113     {
    114         int k=i+1;tot=0;
    115         for3(j,n-sa[i],h[i]+1)
    116         {
    117             while(h[k]>=j)k++;
    118             if(k!=i+1)b[++tot]=k-i;
    119         }
    120         for3(i,tot,1)printf("%d
    ",b[i]);
    121     }        
    122 
    123     return 0;
    124 
    125 }
    View Code
  • 相关阅读:
    Linux开机自动启动ORACLE设置
    Linux Oracle服务启动&停止脚本与开机自启动
    日志挖掘(logminer)
    通过Oracle补充日志,找到锁阻塞源头的SQL
    Oracle Supplemental 补全日志介绍
    修改oracle系统参数spfile导致数据库无法启动解决
    ORA-10997:another startup/shutdown operation of this instance in progress解决方法
    Oracle DataBase单实例使用ASM案例(1)--ASM基本概念
    误删除数据文件,数据库还没有关闭
    2013全国电子设计竞赛进入国赛的项目总结
  • 原文地址:https://www.cnblogs.com/zyfzyf/p/3995513.html
Copyright © 2011-2022 走看看