洛谷P10878 [JRKSJ R9] 在相思树下 III && 单调队列

传送门:P10878 [JRKSJ R9] 在相思树下 III

将军啊,早卸甲,他还在廿二,等你回家……

一道练习单调队列的好题 qwq

题目意思:

很明白了,不再复述。(注意 $ \forall i$ 表示对于任意的 i ,可理解为所有)

思路:

贪心是很明显的,因为我们要让最后的值最大,首先要把小的值删掉。
最后的答案就是进行 m 次操作 1 后的序列中的最小值。
考虑怎样求操作后的序列:

暴力:

枚举 + 判断 绝对TLE

正解

我们发现,操作后的序列中,\(i\) 项的值就是原序列中第 \(i\) 项到第 \(i+m\) 项中的最大值。好像可以用二分吗?
那么我们要做的就是查询每一个 \(i\) 到第 \(i+m\) 的最大值。

  • 蒟蒻首先想到的是线段树。。我承认是线段树水题做多了
    维护线段树的区间最大值,枚举 \(i\)\(1\)\(n-m\) ,每次求 \(i\)\(i+m\) 的最大值,再对所有最大值取最小。。(代码会在后面给)
  • 但是还有更好的做法:单调队列,又称滑动窗口。放一道练手题:P1886 滑动窗口 /【模板】单调队列
    详解请往后翻 qwq
    这样就是个板子了。。。

详解 · 单调队列

队伍内元素保持单调性,保持的方法就是挤掉队尾不能保持单调性的元素。
可以通过双端队列 deque 来实现。
但是 zhx 曾曰过:STL 太慢了
所以给出一份手写的代码:

#include<bits/stdc++.h>

using namespace std;

int n,m;
int a[1001000];
int q[1000100];
int ans1[1000100];
int ans2[1000100];

int main()
{
	cin>>n>>m;
	for(int i=1;i<=n;i++)
	{
		cin>>a[i]; 
	}
	int head=1,tail=0;
	for(int i=1;i<=n;i++)//单调上升
	{
		while(a[i]<a[q[tail]]&&head<=tail)tail--;
		q[++tail]=i;
		while(head<=tail&&q[head]<=i-m) head++;
		if(i>=m)cout<<a[q[head]]<<" ";
	}
	cout<<endl;
	memset(q,0,sizeof(q));
	head=1;tail=0;
	for(int i=1;i<=n;i++)//单调下降
	{
		while(a[i]>a[q[tail]]&&head<=tail)tail--;
		q[++tail]=i;
		while(head<=tail&&q[head]<=i-m) head++;
		if(i>=m)cout<<a[q[head]]<<" ";
	}
	return 0;
}

最终代码:

线段树:只需要有区间查询功能就可以了

#include<bits/stdc++.h>

using namespace std;

#define int long long //不开 long long 见祖宗

int m,n;
int a[1000100];
int minn=LLONG_MAX; //这个地方要用 long long 范围下的最大值,要不然会 WA 掉 1、2、4

struct node{
	int l,r;
	int maxn;
}z[4000100];

node operator+(const node &l,const node &r)
{
	node res;
	res.l=l.l,res.r=r.r;
	res.maxn=max(l.maxn,r.maxn);
	return res;
}

void build(int l,int r,int rt)
{
	if(l==r)
	{
		z[rt].l=z[rt].r=l;
		z[rt].maxn=a[l];
		return;
	}
	int mid=l+r>>1;
	build(l,mid,rt<<1);
	build(mid+1,r,rt<<1|1);
	z[rt]=z[rt<<1]+z[rt<<1|1];
}

int query(int l,int r,int rt,int nowl,int nowr)
{
	if(nowl<=l&&nowr>=r){
		return z[rt].maxn;
	}
	int mid=l+r>>1;
	if (nowl<=mid)
	{
		if (mid<nowr)
			return max(query(l,mid,rt<<1,nowl,nowr),query(mid+1,r,rt<<1|1,nowl,nowr));//这个地方依旧是我的重灾区 qwq
		else
			return query(l,mid,rt<<1,nowl,nowr); 
	}
	else
		return query(mid+1,r,rt<<1|1,nowl,nowr); 
}

signed main()
{
	cin>>n>>m;
	for(int i=1;i<=n;i++){
		cin>>a[i];
	}
	build(1,n,1);
	for(int i=1;i<=n-m;i++)
	{
		int p=query(1,n,1,i,i+m);
		minn=min(minn,p);
	}
	cout<<minn<<endl;
	return 0;
}

单调队列:跟线段树对比来看码量好少

#include<bits/stdc++.h>

using namespace std;

#define int long long

int n,m;
int a[1000100];
int q[1000100];
int ans=LLONG_MAX;

signed main()
{
	cin>>n>>m;
	for(int i=1;i<=n;i++)
	{
		cin>>a[i];
	}
	int head=1,tail=0;
	m+=1;
	for(int i=1;i<=n;i++)//单调下降的序列,求最大值 
	{
		while(a[i]>a[q[tail]]&&head<=tail)tail--;
		q[++tail]=i;
		while(head<=tail&&q[head]<=i-m) head++;
		if(i>=m)ans=min(ans,a[q[head]]);
	}
	cout<<ans<<endl;
 } 
posted @ 2024-08-23 16:17  lazy_ZJY  阅读(8)  评论(0编辑  收藏  举报