题目链接:H. Hopping Around the Array
题目大意:给定一个序列 (a_1,a_2,dots,a_n),在第 (i) 个点可以一步走到 ([i,i+a_i]) 中的任何一个点,多次询问,每一次从 (l) 出发,你可以删掉不超过 (k) 的点,将剩下的点重标号,使得到达 (r) 的步数最少。
(nleq 2 imes 10^4),(kleq 30)。
题解:首先考虑没有删除点应当怎么做,显然我们从第 (i) 个点出发,我们可以到达的点是 ([i,i+a_i]),那么我们假设我们下一步走到的点为 (j(jin [i,i+a_i])),那么 (j) 一定满足 (forall ileq kleq i+a_i,j+a_jgeq k+a_k)。
如果我们删点,那么假设在第 (i) 个点,删除了 (x) 个点,那么显然会走到 (i+a_i+x),否则的话就没有必要删这么多。
那么我们就很容易设计一个状态,设 (f_{i,k,j}) 表示从 (j) 出发,走 (2^i) 步,可以删除不超过 (k) 个点可以走到的最远点。
然后转移的时候我们直接暴力枚举 (k_1,k_2) 将 (max_{l=j}^{f_{i,k_1,j}} f_{i,k_2,l}) 的答案贡献到 (f_{i+1,k_1+k_2,j}) 上,这部分可以通过数据结构来优化。
接下来把所有的询问离线一起处理,设 (ans_{t,i,j}) 表示第 (i) 个询问,删除 (j) 个点,在考虑 (geq 2^t) 步的所有的情况下所能够到达的最靠右的不超过 (r_i) 的点是什么(由我们在题解开头给出的结论,容易得到这样贪心是正确的),初始的时候是 (forall 0leq jleq k_i, ans_{log n,i,j}=l_i)。
然后这一部分中间的转移我们同样可以通过数据结构优化。
时间复杂度为:(O(nklog^2 n + nk^2log n))
代码:
#include <cstdio>
#include <algorithm>
const int Maxn=20000;
const int Maxd=17;
const int Maxk=30;
int n,q;
int nxt[Maxd+1][Maxk+1][Maxn+1];
int f_max[Maxk+1][Maxd+1][Maxn+1];
int a[Maxn+1];
int log_2[Maxn+1];
void init(){
log_2[0]=-1;
for(int i=1;i<=Maxn;i++){
log_2[i]=log_2[i>>1]+1;
}
}
int query_max(int k,int l,int r){
int d=log_2[r-l+1];
return std::max(f_max[k][d][l],f_max[k][d][r-(1<<d)+1]);
}
int ans[Maxn+1];
int pos_r[Maxn+1][Maxk+1];
struct Question{
int l,r,k;
}qu[Maxn+1];
int main(){
init();
scanf("%d%d",&n,&q);
for(int i=1;i<=n;i++){
scanf("%d",&a[i]);
}
for(int i=1;i<=n;i++){
for(int j=0;j<=Maxk;j++){
nxt[0][j][i]=std::min(i+a[i]+j,n);
}
}
for(int i=1;i<=17;i++){
for(int j=1;j<=n;j++){
for(int k=0;k<=Maxk;k++){
f_max[k][0][j]=nxt[i-1][k][j];
}
}
for(int k=0;k<=Maxk;k++){
for(int j=1;(1<<j)<=n;j++){
for(int l=1;l+(1<<j)-1<=n;l++){
f_max[k][j][l]=std::max(f_max[k][j-1][l],f_max[k][j-1][l+(1<<(j-1))]);
}
}
}
for(int k_1=0;k_1<=Maxk;k_1++){
for(int k_2=0;k_1+k_2<=Maxk;k_2++){
for(int j=1;j<=n;j++){
nxt[i][k_1+k_2][j]=std::max(nxt[i][k_1+k_2][j],query_max(k_2,j,nxt[i-1][k_1][j]));
}
}
}
}
for(int i=1;i<=q;i++){
scanf("%d%d%d",&qu[i].l,&qu[i].r,&qu[i].k);
for(int j=0;j<=qu[i].k;j++){
pos_r[i][j]=qu[i].l;
}
if(qu[i].l==qu[i].r){
ans[i]=-1;
continue;
}
}
for(int i=17;i>=0;i--){
for(int j=1;j<=n;j++){
for(int k=0;k<=Maxk;k++){
f_max[k][0][j]=nxt[i][k][j];
}
}
for(int k=0;k<=Maxk;k++){
for(int j=1;(1<<j)<=n;j++){
for(int l=1;l+(1<<j)-1<=n;l++){
f_max[k][j][l]=std::max(f_max[k][j-1][l],f_max[k][j-1][l+(1<<(j-1))]);
}
}
}
for(int j=1;j<=q;j++){
if(ans[j]==-1){
continue;
}
static int np_r[Maxk+5];
for(int k=0;k<=qu[j].k;k++){
np_r[k]=0;
}
for(int k_1=0;k_1<=qu[j].k;k_1++){
for(int k_2=0;k_1+k_2<=qu[j].k;k_2++){
np_r[k_1+k_2]=std::max(np_r[k_1+k_2],query_max(k_2,qu[j].l,pos_r[j][k_1]));
}
}
if(np_r[qu[j].k]<qu[j].r){
ans[j]+=(1<<i);
for(int k=0;k<=qu[j].k;k++){
pos_r[j][k]=np_r[k];
}
}
}
}
for(int i=1;i<=q;i++){
printf("%d
",ans[i]+1);
}
return 0;
}