CodeVS1394 数字串

http://codevs.com/problem/1394/

题意:给定一个长度为n的序列,每个元素都是不大于m的正整数。找出最短的连续数段,满足每一个不大于m的正整数都在其中出现。n和m均不大于200,000。

最简单的想法是,暴力枚举左右端点形成序列,在序列中分别查找每个不大于m的正整数。时间复杂度O(n^3·m)。

可以想到的一个简单优化是,可以枚举左端点,然后不断向右扫描,用一个桶d[]来统计每个不大于m的正整数出现的次数。

则当d[]中的每个元素都大于0时,即每个不大于m的正整数都有出现时,结束当次扫描,修改最小值,选择下一个左端点。时间复杂度O(n^2·m)。

可以想到,每次只修改某单个d[i],即令d[i]增加1,却要把整个数组扫描一次来判断,是很浪费时间的。我们可以令cnt为d[]中值大于0的元素个数,且初始化为0。

每次令d[i]增加1后,若d[i]的变化是从0到1,则令cnt增加1。当cnt=m时就知道当前数段已经满足要求了,不必次次把d[]扫描一次。时间复杂度O(n^2+nm)。

但是本题要求一个线性的做法,每次选择好左端点然后从原地开始向右扫描,显然是二次方的。是否一定要从原地开始扫描呢?

令f[i]为当左端点为i时,最小的使得数段[i,j]满足要求的j。如n=5,m=3且序列为1,3,3,2,1,当左端点为1时向右扫描,直到扫描到第4个数,才能使得当前数段1,3,3,2满足要求。

而以2为左端点则要扫描到第5个数才可以。因此定义f[1]=4,f[2]=5,同理f[3]=5,而f[4]和f[5]为无穷大。则只需要在所有f[i]-i+1中找一个最小值即可。

应该注意到,f[i]关于i是单调的,当i增加时,f[i]不会减少。如果有i<j且f[j]<f[i],由于数段[i,f[j]]包含数段[j,f[j]],一定也是满足要求的,而数段[i,f[i]]比这还大,不可能是最小的。

当i=1时,可以进行朴素的扫描,即从原地开始。当i=2时,则只要从f[1]开始即可。更一般地,当左端点为i时,只要从f[i-1]开始即可,大大节省了时间。

但是,d[]和cnt要怎么维护呢?只需要在左端点i移动至i+1时,将d[i]减去1即可。如果d[i]的变化是从1到0,则令cnt减去1。右端点移动时,就按原来添加元素的方法去做即可。

代码:

#include<cstdio>
int s[200005]={0},S=0;
int a[200005],n,m,i,j=1,k=1<<30,t;
int main(){
    scanf("%d%d",&n,&m);
    for(i=1;i<=n;i++) scanf("%d",&a[i]);
    i=0;
    while((S<m)&&(i<=n)){
    	i++;
    	s[a[i]]++;if(s[a[i]]==1) S++;
	}
	if(i>n){
		printf("NO");
		return 0;
	}
	k=i;
    while(i<=n){
        s[a[i]]++;if(s[a[i]]==1) S++;
        while((S>m)||(S=m)&&(s[a[j]]>1)){
        	s[a[j]]--;if(s[a[j]]==0) S--;
        	j++;
		}
        if(i-j+1<k) k=i-j+1;
        i++;
    }
    printf("%d\n",k);
    return 0;
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值