尺取法用来干什么?
尺取法一般用来算有关连续子序列有关的问题。注意,是连续!!!
什么是尺取法?
顾名思义,就是像尺子一样一段一段的取连续子序列,根据题目的不同要求做不同的判断条件(例如有时候会是循环连续子序列)
核心思想
取一个l,一个r做下标的标记,假如我要算一个长度为n的数组的连续子序列的和满足小于等于总和的一半的最短长度,一开始让l和r都等于0,让r不断右移直到大于总和的一半,判断一次当前长度-1与之前长度哪个更小,然后l也开始右移,直到 l到r的和小于等于总和的一半,判断当前长度与之前长度哪个更小,再重复r右移…
在一次次的对满足条件的连续子序列最短长度的更新下,我们最后能够得到答案,那么,要循环多少次这个运行?
我们假设是最坏的条件,r每右移一次就大于总和一半,l每左移一次就小于等于总和一半(像1 1 或者1 1 1 这样的数列),那我们就要循环2次(1 1数列),所以在n次循环后我们一定能得到答案,当然会有一定的多余时间损耗,但是因为尺取法比直接暴力枚举区间效率高很多,尤其是数据量大的时候,所以尺取法会更加好用。
模板代码(我自己的qaq)
//最短连续子序列的长度 (和小于总和一半)
#include<bits/stdc++.h>
using namespace std;
int main(){
int n,sum=0,ans=0,sum2=0;
scanf("%d",&n);
int a[n];
ans=n;//给ans一个较大的数,能让它在第一次比较中得到较小的数
for(int i=0;i<n;i++){
scanf("%d",&a[i]);
sum+=a[i];
}
int l=0,r=0;
for(int i=0;i<n;i++){
while(r<n&&sum2<=sum/2){
sum2+=a[r++];
}
if(sum2<=sum/2){
break; //数组加到最后也没能大于sum/2,可以直接跳出去了
}
ans=min(ans,r-l);//原本应该是r-l+1(例如0到2有3个数),但因为此时刚好大于sum/2,所以要减掉1
while(l<n&&sum2>sum/2){
sum2-=a[l++];
}
ans=min(ans,r-l+1);
}
printf("%d",ans);
return 0;
}
附带一个大佬眼中的签到题
这个是循环子序列最大值的查找,附上我的ac代码
#include<bits/stdc++.h>
using namespace std;
int main(){
int n,sum=0;
scanf("%d",&n);
int a[n];
for(int i=0;i<n;i++){
scanf("%d",&a[i]);
sum+=a[i];
}
int l=0,r=0,ans=0,a1=0;
for(int i=0;i<n;i++){
while(a1<sum/2){
a1+=a[r%n];
r++;
}
while(a1>sum/2){
a1-=a[l%n];
l++;
}
ans=max(a1,ans);
if(ans==sum/2){
break;
}
}
ans=min(ans,sum-ans);
printf("%d",ans);
return 0;
}