尺取法

尺取法用来干什么?

尺取法一般用来算有关连续子序列有关的问题。注意,是连续!!!

什么是尺取法?

顾名思义,就是像尺子一样一段一段的取连续子序列,根据题目的不同要求做不同的判断条件(例如有时候会是循环连续子序列)

核心思想

取一个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;
}

附带一个大佬眼中的签到题

蓝桥ADV-381

这个是循环子序列最大值的查找,附上我的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;
} 
  • 2
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值