BZOJ2086: [Poi2010]Blocks

题解:

想了想发现只需要求出最长的一段平均值>k即可。

平均值的问题给每个数减去k,判断是否连续的一段>0即可。

然后我们发现如果i<j 且 s[i]<s[j],那么 j 对于l>j不会比i 优。

那我们就可以维护一个单调的s[i],然后对于每个l去二分出它的答案。

但这样会T。

再次考虑单调性。

如果 l>j,且s[l]>s[i],那么我们的答案最小也是l-i,所以j必须取比i小的s才有可能更新答案。然后就可以两个指针扫一遍了。

好吧 我承认我口胡。。。

代码:

 1 #include<cstdio>
 2 #include<cstdlib>
 3 #include<cmath>
 4 #include<cstring>
 5 #include<algorithm>
 6 #include<iostream>
 7 #include<vector>
 8 #include<map>
 9 #include<set>
10 #include<queue>
11 #include<string>
12 #define inf 1000000000
13 #define maxn 1000000+5
14 #define maxm 1000000+5
15 #define eps 1e-10
16 #define ll long long
17 #define pa pair<int,int>
18 #define for0(i,n) for(int i=0;i<=(n);i++)
19 #define for1(i,n) for(int i=1;i<=(n);i++)
20 #define for2(i,x,y) for(int i=(x);i<=(y);i++)
21 #define for3(i,x,y) for(int i=(x);i>=(y);i--)
22 #define for4(i,x) for(int i=head[x],y=e[i].go;i;i=e[i].next,y=e[i].go)
23 #define mod 1000000007
24 using namespace std;
25 inline ll read()
26 {
27     ll x=0,f=1;char ch=getchar();
28     while(ch<‘0‘||ch>‘9‘){if(ch==‘-‘)f=-1;ch=getchar();}
29     while(ch>=‘0‘&&ch<=‘9‘){x=10*x+ch-‘0‘;ch=getchar();}
30     return x*f;
31 }
32 int n,m,q[maxn];
33 ll a[maxn],b[maxn];
34 int main()
35 {
36     freopen("input.txt","r",stdin);
37     freopen("output.txt","w",stdout);
38     n=read();m=read();
39     for1(i,n)a[i]=read();
40     while(m--)
41     {
42         ll x=read();int r=0;
43         q[r=1]=b[0]=0;
44         for1(i,n)
45         {
46           b[i]=b[i-1]+a[i]-x;
47           if(b[i]<b[q[r]])q[++r]=i;
48         }
49         int ans=0;
50         for(int i=n;i>0&&r;i--)
51         {
52             while(r&&b[i]>=b[q[r]])r--;
53             r++;
54             if(b[i]>=b[q[r]])ans=max(ans,i-q[r]);
55         }
56         printf("%d",ans);if(m)printf(" ");else printf("\n");
57     }
58     return 0;
59 }

2086: [Poi2010]Blocks

Time Limit: 20 Sec  Memory Limit: 259 MB
Submit: 199  Solved: 91
[Submit][Status]

Description

给出N个正整数a[1..N],再给出一个正整数k,现在可以进行如下操作:每次选择一个大于k的正整数a[i],将a[i]减去1,选择a[i-1]或a[i+1]中的一个加上1。经过一定次数的操作后,问最大能够选出多长的一个连续子序列,使得这个子序列的每个数都不小于k。
总共给出M次询问,每次询问给出的k不同,你需要分别回答。

Input

第一行两个正整数N (N <= 1,000,000)和M (M <= 50)。
第二行N个正整数,第i个正整数表示a[i] (a[i] <= 10^9)。
第三行M个正整数,第i个正整数表示第i次询问的k (k <= 10^9)。

Output

共一行,输出M个正整数,第i个数表示第i次询问的答案。

Sample Input

5 6
1 2 1 1 5
1 2 3 4 5 6

Sample Output

5 5 2 1 1 0

时间: 2024-12-19 04:29:43

BZOJ2086: [Poi2010]Blocks的相关文章

[bzoj2086][Poi2010]Blocks_单调栈_双指针

Blocks bzoj-2086 Poi-2010 题目大意:题目链接. 注释:略. 想法:首先,不难发现,如果连续的一段数的平均值不小于输入的k的话,这段数是满足题意的. 所以,我们再次简化一下:将每个数都减去k,即求极大区间,使得区间和为正. 将所有数的前缀和自尾至头压进单调栈,然后左指针遍历1->n,右指针在单调栈上扫即可. 最后,附上丑陋的代码... ... #include <iostream> #include <cstdio> #include <cstr

BZOJ 2086: [Poi2010]Blocks

2086: [Poi2010]Blocks Time Limit: 20 Sec  Memory Limit: 259 MBSubmit: 494  Solved: 222[Submit][Status][Discuss] Description 给出N个正整数a[1..N],再给出一个正整数k,现在可以进行如下操作:每次选择一个大于k的正整数a[i],将a[i]减去1,选择a[i-1]或a[i+1]中的一个加上1.经过一定次数的操作后,问最大能够选出多长的一个连续子序列,使得这个子序列的每个数

bzoj 2086 [Poi2010]Blocks 单调栈

[Poi2010]Blocks Time Limit: 20 Sec  Memory Limit: 259 MBSubmit: 788  Solved: 356[Submit][Status][Discuss] Description 给出N个正整数a[1..N],再给出一个正整数k,现在可以进行如下操作:每次选择一个大于k的正整数a[i],将a[i]减去1,选择a[i-1]或a[i+1]中的一个加上1.经过一定次数的操作后,问最大能够选出多长的一个连续子序列,使得这个子序列的每个数都不小于k.

栈和队列小刷

栈和队列 bzoj1012// JSOI2008 最大数 maxnumber 详见博客:https://www.cnblogs.com/tyner/p/11267630.html bzoj2086 [Poi2010]Blocks https://www.luogu.org/problem/P3503 题意 给出 N 个正整数 a[1...N],再给出一个正整数 k,现在可以进行 如下操作: 每次选择一个大于 k 的正整数 a[i],将 a[i] 减去 1,选择a[i - 1] 或 a[i + 1

「POI2010」积木 Blocks

单调栈 可以发现子序列的平均数不小于k就可以使子序列的每个数都不小于k 将每个数减去k,则子序列之和非负即可 记 si = a1 + a2 +....+ ai - k*i 考虑序列的两个端点 l, r 对于 l: l1 < l2 && sl1 <=  sl2, 则l1 比 l2 优 对于 r:   r1 < r2 && sr1 <= sr2 则r2 比 r1 优 从0到n 将可能最优 l 加入单调栈中,可以发现栈是单调递增的 从n到1 的可能最优 r

单调队列 BZOJ 2096 [Poi2010]Pilots

2096: [Poi2010]Pilots Time Limit: 30 Sec  Memory Limit: 162 MBSubmit: 819  Solved: 418[Submit][Status][Discuss] Description Tz又耍畸形了!!他要当飞行员,他拿到了一个飞行员测试难度序列,他设定了一个难度差的最大值,在序列中他想找到一个最长的子串,任意两个难度差不会超过他设定的最大值.耍畸形一个人是不行的,于是他找到了你. Input 输入:第一行两个有空格隔开的整数k(0

【Code::Blocks】windows 环境下编译 Code::Blocks(已修正)

Code::Blocks 在2012-11-25发布了最新的12.11版本,相比上一个版本(10.05),Code::Blocks 进行了许多改进和更新(Change log). 引用 Wikipedia:Code::Blocks: Code::Blocks是一个免费.开源.跨平台的IDE,使用C++开发,并且使用wxWidgets做为GUI函式库.Code::Blocks使用了插件架构,其功能可以使用插件自由地扩充.目前, Code::Blocks主要针对开发C/C++程式而设计. Code:

Using ffmpeg libraries in Code::Blocks

I am digging into a project that needs to use some open source video codec library. ffmpeg is very powerful video codec library, which I decided to use in my project. I compiled and installed the ffmpeg library, then the problem comes, how to use the

BZOJ 2080: [Poi2010]Railway 双栈排序

2080: [Poi2010]Railway Time Limit: 10 Sec  Memory Limit: 259 MBSubmit: 140  Solved: 35[Submit][Status][Discuss] Description 一个铁路包含两个侧线1和2,右边由A进入,左边由B出去(看下面的图片) 有n个车厢在通道A上,编号为1到n,它们被安排按照要求的顺序(a1,a2,a3,a4....an)进入侧线,进去还要出来,它们要按照编号顺序(1,2,3,4,5....n)从通道B