codeforces 338 E. Optimize! (线段树维护最小前缀和)
题意:题意是由伪代码给出的。。手算模拟了一下(noip初赛即视感),题意大概是说,给出两个数组a和b,a数组长度为n,b数组长度为len,然后从a中截取连续的len个元素,称为数组s,如果存在一种方法使得s中元素和b中的元素一一对应且每组和都大于等于h,则称这个s是合法的。现在问a中有多少个合法的s。 具体来说,对于样例 5 2 10 5 3 1 8 5 5 7
s={8,5}和s={5,7}是有解的。 因为对于前者8+3>=10并且5+5>=10,对于后者5+5>=10,7+3>=10,因此答案为2 思路:
最重要的一部是,我们先将b排序,然后维护一个函数f[i],在排序后的b中找到一个最小的j,满足a[i]+b[j]>=h,然后让f[i]=j?
f[i]直接二分就可以得到。。
因为f[i]的值是使得a[i]满足条件的最小的b值。
然后我们观察发现。
f[i]大于等于len的元素个数最多有1个,不然无解。
f[i]大于等于len-1的元素个数最多有2个,不然无解。
f[i]大于等于i的元素个数最多由len+1-i个,不然无解。
设g[i]=len+1-i-sum[i],sum[i]=y[i]+y[i+1]+...+y[len]。
如果某个s中,对于所有的i,都有g[i]>=0,那么这个s就是合法的。
实际上我们没有必要去查询每个g[i],只要g[i]中最小的大于等于0,那么这个s就是合法的。
现在的问题就变成了动态维护一段长度为len的最小后缀和。
我的思路是整体覆盖,
当a数组的区间从[l,r]到[l+1,r+1]的时候,增加了a[r+1],假设f[r+1]=p,那么用线段树更新区间[1,p],都增加1. 减少了a[l],假设f[l]=q,那么同样用线段树维护,使得区间[1,q]都减少1.
讲道理应该也可以做吧。。。
不过被羊神@sheep教育说其实单点更新就可以。。。
于是去学了下动态维护区间最大子段和的线段树做法: bzoj 1756解题报告hit oj 2687 解题报告
然后单点更新维护最大后缀和就很容易了。
类似的做法,我们也维护一个最小前缀和。初始化的时候把1..len赋值成-1.
然后每次移动区间,从【l,r】移动到[l+1,r+1]的时候,将f[r+1]添加1,将f[l]减少1.
这样如果某个s中的任意前缀和都是等于0的,那么这个s就是合法的。
因此我们只需要维护最小前缀和。
代码是维护的最小前缀和
1/* ***********************************************
2Author :111qqz
3Created Time :Fri 16 Sep 2016 05:13:38 PM CST
4File Name :code/cf/problem/338E.cpp
5************************************************ */
6#include <cstdio>
7#include <cstring>
8#include <iostream>
9#include <algorithm>
10#include <vector>
11#include <queue>
12#include <set>
13#include <map>
14#include <string>
15#include <cmath>
16#include <cstdlib>
17#include <ctime>
18#define fst first
19#define sec second
20#define lson l,m,rt<<1
21#define rson m+1,r,rt<<1|1
22#define ms(a,x) memset(a,x,sizeof(a))
23typedef long long LL;
24#define pi pair < int ,int >
25#define MP make_pair
26using namespace std;
27const double eps = 1E-8;
28const int dx4[4]={1,0,0,-1};
29const int dy4[4]={0,-1,1,0};
30const int inf = 0x3f3f3f3f;
31const int N=150005;
32int n,len,h;
33int a[N],b[N],f[N];
34struct Tree
35{
36 int mn;//最小前缀和
37 int sum;//区间和
38}tree[N<<2];
39void PushUp( int rt)
40{
41 tree[rt].sum = tree[rt<<1].sum + tree[rt<<1|1].sum;
42 tree[rt].mn = min(tree[rt<<1].mn,tree[rt<<1].sum+tree[rt<<1|1].mn);
43}
44void build( int l,int r,int rt)
45{
46 if (l==r)
47 {
48 if (l<=len)
49 tree[rt].sum = tree[rt].mn = -1;
50 return;
51 }
52 int m = (l+r)>>1;
53 build(lson);
54 build(rson);
55 PushUp(rt);
56}
57void update( int p,int sc,int l,int r,int rt)
58{
59 if (l==r)
60 {
61 tree[rt].sum += sc;
62 tree[rt].mn +=sc;
63 return;
64 }
65 int m = (l+r)>>1;
66 if (p<=m) update(p,sc,lson);
67 else update(p,sc,rson);
68 PushUp(rt);
69}
70int bin( int x)
71{
72 int l = 1;
73 int r = len;
74 int res = len+1; //无穷。
75 while (l<=r)
76 {
77 int mid = (l+r)>>1;
78 if (b[mid]+x>=h)
79 {
80 r = mid-1;
81 res = mid;
82 }
83 else
84 {
85 l = mid+1;
86 }
87 }
88 return res;
89}
90int main()
91{
92 #ifndef ONLINE_JUDGE
93 freopen("code/in.txt","r",stdin);
94 #endif
95 ios::sync_with_stdio(false);
96 ms(tree,0);
97 cin>>n>>len>>h;
98 int siz = len+1;
99 for ( int i = 1 ;i <= len ; i++) cin>>b[i];
100 sort(b+1,b+len+1);
101 for ( int i = 1 ; i <= n ; i++)
102 {
103 int x;
104 cin>>x;
105 a[i] = bin(x);
106 }
107 build(1,siz,1);
108 for ( int i = 1 ; i < len ; i++) update(a[i],1,1,siz,1);
109 int ans = 0 ;
110 for ( int i = len ; i <= n ; i++)
111 {
112 update(a[i],1,1,siz,1);
113 if (tree[1].mn==0) ans++;
114 update(a[i-len+1],-1,1,siz,1);
115 }
116 cout<<ans<<endl;
117 #ifndef ONLINE_JUDGE
118 fclose(stdin);
119 #endif
120 return 0;
121}