[ACM_其他] 总和不小于S的连续子序列的长度的最小值——尺缩法

Description:

给定长度为n的整数数列,A[0],A[1],A[2]….A[n-1]以及整数S,求出总和不小于S的连续子序列的长度的最小值。如果解不存在,则输出0。

 

Input:

输入数据有多组,每组数据第一行输入n,S, (10<n<10^5,S<10^8)第二行输入A[0],A[1],A[2]….A[n-1] ( 0<A[i]≤10000),处理到文件结束。所有输入数据都在int范围之内。

Output:

每组数据输出结果占一行。

Sample Input:

10 15

5 1 3 5 10 7 4 9 2 8

Sample Output:

2


>>>>>>>>>这题有一个技巧,叫做尺缩法
-----------------------------------------------------------------------------------------------------------------------

(1)   设置两个指针s和t,一开始都指向数列第一个元素,此外sum=0,res=0;

(2)   只要sum<S,就将sum增加一个元素,t加1;

(3)   直到sum>=S,更新res=min(res,t-s);

(4)   将sum减去一个元素,s加1,执行(2)。

上述流程反复地推进区间的开头和末尾,来求取满足条件的最小区间。

----------------------------------------------------------------------------------------------------------------------------------------------------------

#include<iostream>

using namespace std;

int min(int a,int b){if(a>b)return b;return a;}

int main(){

    int n,S,A[100005];

    int i,j,k;

    while(cin>>n>>S){

        //输入数据部分

        for(int i=0;i<n;i++)cin>>A[i];

        //尺缩法计算部分

        int res=n+1,s=0,t=0,sum=0;

        for(;;){

            while(t<n && sum<S)sum+=A[t++];

            if(sum<S)break;

            res=min(res,t-s);

            sum-=A[s++];

        }

        if(res>n)res=0;

        //输出结果

        cout<<res<<'\n';

    }

    return 0;

}
View Code

 

>>>>>>>>> 同样也可以用用lower_bound:

   --------------------------------------------------------------------------------------------------------------------------------------------------------

   lower_bound(ForwardIterator first,ForwardIterator last, const Type &value,Compare comp );

 lower_bound()返回一个 iterator 它指向在[first,last)标记的有序序列中可以插入value,而不会破坏容器顺序的第一个位置,而这个位置标记了一个大于等于value 的值。

   例如,有如下序列:

   ia[]={12,15,17,19,20,22,23,26,29,35,40,51};
   用值21调用lower_bound(),返回一个指向22的iterator。用值22调用lower_bound(),也返回一个指向22的iterator。根据comp进行排序和比较。
  注意:::调用lower_bound之前必须确定序列为有序序列,否则调用出错。函数lower_bound()在first和last中的前闭后开区间进行二分查找,返回大于或等于    val的第一个元素位置。如果所有元素都小于val,则返回last的位置。
   --------------------------------------------------------------------------------------------------------------------------------------------------------
   所以这里我们可以这样处理:

   我们可以计算出sum(i)=a0+a1+...+ai。那么sum(t)-sum(s)=as+a(s+1)+...a(t-1)。这样我们可以实现先求出一个sum(n)。sum(n)-sum(i)=s。我们只需要去筛选sum(i)。这样可以用二分搜索的方法快速求出最小的长度。

int t=lower_bound(sum+i,sum+n,sum[i]+s)-sum;求出来的是从ai到at(i<=t<=n)和比s小的最小值的下标。

 1 #include<algorithm>

 2 using namespace std;

 3 int min(int a,int b){if(a>b)return b;return a;}

 4 int n,s;

 5 int a[100005];

 6 int sum[100005];

 7 int main(){

 8     while(cin>>n>>s){

 9         //sum归0;

10         memset(sum,0,sizeof(sum));

11         //输入并计算sum;

12         for(int i=0;i<n;i++){

13             cin>>a[i];

14             sum[i+1]=sum[i]+a[i];

15         }

16         //运算求解输出;

17         if(sum[n]<s){

18             cout<<"0"<<endl;

19         }

20         else{

21             int ret=n;

22             for(int i=0;sum[i]+s<=sum[n];i++){          

23                 int t=lower_bound(sum+i,sum+n,sum[i]+s)-sum;

24                 ret=min(ret,t-i);

25             }

26             cout<<ret<<endl;

27         }

28     }

29     return 0;

30 }
View Code

 

>>>>>>>>附加:折半查找法:

 1 int lowerBound(int array[],int left,int right,int value)

 2 {

 3         int mid=0,half=0;

 4         int len=(right+left+1)/2;

 5         while(len>0)

 6         {

 7                 half=len>>1;          //数据长度折半

 8                 mid=left+half;        //计算中点

 9                 if (array[mid]<value) //调整总长与起点

10                 {

11                         len=len-half-1;

12                         left=mid+1;

13                 }

14                 else

15                         len=half;

16         }

17         return left;

18 }
lower_bound版本一:
 1 int lowerBound(int array[],int left,int right,int value)

 2 {

 3         int mid=0;

 4         while(left<right)

 5         {

 6                 mid=(right+left)/2;       //计算中点

 7                 if (array[mid]<value)     //调整起点或者终点

 8                         left=mid+1;

 9                 else

10                         right=mid;

11         }

12         return right;

13 }
lower_bound版本二:
 1 int upperBound(int array[],int left,int right,int value)

 2 {

 3         int mid=0,half=0;

 4         int len=(right+left+1)/2;

 5         while(len>0)

 6         {

 7                 half=len>>1;         //长度折半

 8                 mid=left+half;       //计算中点

 9                 if (array[mid]>value) //调整长度与起点

10                         len=half;

11                 else

12                 {

13                         len=len-half-1;

14                         left=mid+1;

15             }

16         }

17         return left;

18 }
upper_bound版本一:
 1 int upperBound(int array[],int left,int right,int value)

 2 {

 3         int mid=0;

 4         while(left<right)

 5         {

 6                 mid=(right+left)/2;    //计算中点

 7                 if (array[mid]>value)  //调整起点或者终点

 8                         right=mid;

 9                 else

10                         left=mid+1;

11         }

12         return right;

13 }
upper_bound版本二:
 1 int binarySearch(int array[],int left,int right,int value)

 2 {

 3         int mid;

 4         while(left<=right)

 5         {

 6                 mid=(left&right)+((left^right)>>1);      //防止溢出

 7                 if(array[mid]==value)

 8                         return mid;

 9                 else if (array[mid]<value)

10                         left=mid+1;

11                 else

12                         right=mid-1;

13         }

14         return -1;

15 }
折半查找:

 

 

 

你可能感兴趣的:(ACM)