7.3总结
得分情况
估分:60+40+100=200
实际:60+40+30=130
Rank 9
萎的一批
好歹把T1T2的暴力分拿满了
T1
比赛的时候想了很久没有思路,就打暴力把人一个个塞进去,拿了暴力分60分
正解:
操作一:
首先,放人进去的时候,优先级是固定的。所以我们可以用后序遍历把优先级搞出来,用一个堆来维护哪些位置是空的,每次放人的时候就从堆里面找一个位置,删人的时候放一个位置进堆里面。(用线段树也可以搞,我就是用线段树打的)
操作二:
我们已经知道哪些点有人,那么把一个人删去时,就用倍增找到他最上面一个有人的节点,把它删去就好了。(用树链剖分也可以搞,cgh打了树剖)
T2
比赛的时候想了很久也没什么思路,就把40分暴力分给拿到了。
正解
首先考虑一种暴力:设f[i][0/1]表示前面的数与i进行位运算,结果的最大值和方案数。这样子做插入是O(\(2^{16}\))的,询问是O(1)的。
怎么平衡一下插入和询问的时间复杂度呢?
运用广义平衡规划的思想,类似折半搜索一样,设f[i][j][0/1]表示前面的数中,前八位为i,与某个后八位为j的数进行位运算,后八位结果的最大值和方案数。
这样一来,插入时i是固定的,我们只用枚举j来更新f;而询问时j是固定的,只用枚举i来更新ans。
好巧妙的思想啊
T3
谴责出题人,暴力随便水过。
我以为它是满足二分性的,结果又TLE有WA爆炸了。
早知道就打暴力算了。
正解
首先,把读入的串分段,同一段里面满足每个数%d都是同一个值
由于不能有重复的数,所以要求一个maright[l]来表示最大右端点使a[l~maright[l]]中没有相同的数。
用哈希或map随便搞搞就好了
然后\(n^2\)暴力就切了哈哈
那么为了避免负数等情况,我们处理一下数据。对于一段每个数a[x],令a[x]=(a[x]-min)/d+1,min即为这一段的最小值。
当l,r为左右端点时,满足\(max\{a_{l...r}\}-min\{a_{l...r}\}+1\leq r-l+1+k\)即可更新答案。
考虑枚举左端点,找最大的右端点
设ma[r]表示当左端点为l时的\(max\{a_{l...r}\}\),mi[r]表示当左端点为l时的\(min\{a_{l...r}\}\)
化一下式子变成\(ma[r]-mi[r]-r\leq k-l\),右边只与l有关。
设\(b[r]=ma[r]-mi[r]-r\),用线段树维护b,那么只用找出最大的r使\(b[r]\leq k-l\)且r<=maright[l]
倒着枚举l,这样每一个ma[r]和mi[r]就只会单调上升或下降。其实ma与mi两个数组本身也是从左到右单调的。
那么当我们把l-1时,从l往后某一段ma就会由 一个小于a[l-1]的数 变为a[l-1]。但是我们不能直接用区间赋值,因为那样的话,由于那一段区间的ma[r]和mi[r]可能不同,所以没法更新区间b[r]的最小值。但是用区间加法就可以了。所以要打两个单调栈维护相同的一块块ma[r]和mi[r]。如果这个区间的ma[r]都相同(其实就等于a[r])且小于a[l],更新标记的时候给这个区间加上a[l]-ma[r];如果这个区间的mi[r](其实就等于a[r])都相同且大于a[l],更新标记的时候给这个区间加上mi[r]-a[l]。//这个地方打反了正负号调了一个小时。
由于线段树里面我们维护的是b[r]的最小值,所以最后用线段树的区间查询找l~n里面最靠右的满足b[r]\(\leq\)k-l并且r<=maright[l]的点就是以l为左端点的答案。
(这几天做过的最难的题目)
(其实也没多难,暴力就切了嘛233)
//for the moon
#include<cstdio>
#include<cstring>
#include<algorithm>
#define mod 19260817
using namespace std;
struct qy
{
int mi,add;
};
int n,k,d,i,j,st,en,t;
int a[200005],maright[200005];
int ans1,ans2;
int hash[mod+5];
int ma[200005],mi[200005];
int z1[200005],z2[200005];
qy tree[800005];
void pushdown(int k,int l,int r)
{
if (l!=r)
{
tree[k*2].add+=tree[k].add;
tree[k*2+1].add+=tree[k].add;
tree[k*2].mi+=tree[k].add;
tree[k*2+1].mi+=tree[k].add;
}
tree[k].add=0;
}
void update(int k,int l,int r)
{
tree[k].mi=2100000000;
int mid=(l+r)/2;
if (l<=mid) tree[k].mi=min(tree[k].mi,tree[k*2].mi);
if (mid+1<=r)
tree[k].mi=min(tree[k].mi,tree[k*2+1].mi);
}
void insert(int k,int l,int r,int x)
{
pushdown(k,l,r);
if (l==r)
{
tree[k].add=0;
tree[k].mi+=-x;
}
else
{
int mid=(l+r)/2;
if (x<=mid) insert(k*2,l,mid,x);
else insert(k*2+1,mid+1,r,x);
update(k,l,r);
}
}
void add(int k,int l,int r,int x,int y,int z)
{
pushdown(k,l,r);
if ((l<=y)&&(r>=x))
{
if ((l>=x)&&(r<=y))
{
tree[k].add+=z;
tree[k].mi+=z;
return;
}
int mid=(l+r)/2;
add(k*2,l,mid,x,y,z);
add(k*2+1,mid+1,r,x,y,z);
update(k,l,r);
}
}
int query(int k,int l,int r,int x,int y,int z)
{
pushdown(k,l,r);
if ((l<=y)&&(r>=x))
{
int mid=(l+r)/2;
if ((l>=x)&&(r<=y))
{
if (l==r) return l;
else
{
if (tree[k*2+1].mi<=z) return query(k*2+1,mid+1,r,x,y,z);
else return query(k*2,l,mid,x,y,z);
}
}
int tt=0;
if (tree[k*2+1].mi<=z)
tt=query(k*2+1,mid+1,r,x,y,z);
if (tt!=0) return tt;
else return query(k*2,l,mid,x,y,z);
}
else
return 0;
}
void clear(int k,int l,int r)
{
tree[k].mi=tree[k].add=0;
int mid=(l+r)/2;
if (l!=r)
{
clear(k*2,l,mid);
clear(k*2+1,mid+1,r);
}
}
void does(int st,int en)
{
int mii=2000000000;
int l,r,ll,rr;
for (i=st;i<=en;i++)
mii=min(mii,a[i]);
for (i=st;i<=en;i++)
a[i]=(a[i]-mii)/d+1;
z1[0]=z2[0]=0;
for (l=en;l>=st;l--)
{
ma[l]=mi[l]=a[l];
insert(1,st,en,l);
while ((z1[0]>=1)&&(a[z1[z1[0]]]<=a[l]))
{
ll=z1[z1[0]];
rr=z1[z1[0]-1]-1;
if (z1[0]==1) rr=en;
add(1,st,en,ll,rr,a[l]-a[ll]);
z1[0]--;
}
z1[++z1[0]]=l;
while ((z2[0]>=1)&&(a[z2[z2[0]]]>=a[l]))
{
ll=z2[z2[0]];
rr=z2[z2[0]-1]-1;
if (z2[0]==1) rr=en;
add(1,st,en,ll,rr,a[ll]-a[l]);
z2[0]--;
}
z2[++z2[0]]=l;
r=query(1,st,en,l,maright[l],k-l);
if (r-l>=ans2-ans1)
{
ans1=l;
ans2=r;
}
}
clear(1,st,en);
}
int js(long long x)
{
return (long long)x%mod*x%mod;
}
int main()
{
freopen("read.in","r",stdin);
scanf("%d%d%d",&n,&k,&d);
for (i=1;i<=n;i++)
{
scanf("%d",&a[i]);
a[i]=a[i]+1000000000;
}
maright[n]=n;
hash[js(a[n])]=n;
for (i=n-1;i>=1;i--)
{
t=hash[js(a[i])];
maright[i]=maright[i+1];
if (t!=0)
maright[i]=min(t-1,maright[i]);
hash[js(a[i])]=i;
}
st=1;
ans1=ans2=1;
for (i=2;i<=n;i++)
{
if (a[i]%d!=a[i-1]%d)
{
en=i-1;
does(st,en);
st=i;
}
}
does(st,n);
printf("%d %d",ans1,ans2);
return 0;
}