首页 > 其他 > 详细

E - Subsequence

时间:2015-10-28 19:21:12      阅读:235      评论:0      收藏:0      [点我收藏+]

LA 3029

A sequence of N positive integers (10 < N < 100 000), each of them less than or equal 10000, and a positive integer S (S < 100 000 000) are given. Write a program to find the minimal length of the subsequence of consecutive elements of the sequence, the sum of which is greater than or equal to S.

Input

Many test cases will be given. For each test case the program has to read the numbers N and S, separated by an interval, from the first line. The numbers of the sequence are given in the second line of the test case, separated by intervals. The input will finish with the end of file.

Output

For each the case the program has to print the result on separate line of the output file.

Sample Input

10 15

5 1 3 5 10 7 4 9 2 8

5 11

1 2 3 4 5

Sample Output

2

3

 

题意:给出n个正整数,要求从中选出最短的一个子序列,这个子序列的和要大于或者等于s

思路:这道题主要难度在于时间,数据范围是十的五次方,如果采用传统的O(n^2)算法恐怕会力不从心,因此要想到改变枚举方式,只枚举子序列的终点?

使用前缀合sum[i],那么最开始是设j=0,那么开始枚举sum[i]-sum[j],如果找到一个符合条件的子序列,就尝试逐渐增加j,最后也无需把j置零,因为sum数列是递增的.如果前面有sum[i]-sum[j]符合条件,那么任何大于i的k都符合sum[k]-sum[j]>=s

 

技术分享
#include <cstdio>
#include <iostream>
using namespace std;
const int maxn=1e5+10;
int sum[maxn];
int main()
{
    int n,k;
    while(scanf("%d%d",&n,&k)!=EOF)
    {
     sum[0]=0;
     int a;
     for(int i=1;i<=n;i++) {scanf("%d",&a);sum[i]=sum[i-1]+a;}
     int j=0;
     int minx=1e9;
     for(int i=0;i<=n;i++)
     {
       if(sum[i]-sum[j]<k) continue;
       while(sum[i]-sum[j]>=k) j++;
       minx=min(i-j+1,minx);
     }
     printf("%d\n",minx==1e9?0:minx);
    }
    return 0;
}
View Code

这段程序的时间复杂度为O(n),因为外层虽然i从1循环到n,而对于每个i都不可能有从1到n的j循环,顶多就是所有i内的j循环加起来等于n次而已

E - Subsequence

原文:http://www.cnblogs.com/zsyacm666666/p/4918250.html

(0)
(0)
   
举报
评论 一句话评论(0
关于我们 - 联系我们 - 留言反馈 - 联系我们:wmxa8@hotmail.com
© 2014 bubuko.com 版权所有
打开技术之扣,分享程序人生!