设f[i]为[1,i]分组的最优解,则

f[i]=max(f[j]+1),max(c[j+1],c[j+2],...,c[i-1],c[i])<=i-j<=min(d[j+1],d[j+2],...,d[i-1],d[i])

设g[i]=min(j),i-j<=min(d[j+1],d[j+2],...,d[i-1],d[i])

容易发现g[i]单调不下降,可以通过线段树$O(n\log n)$预处理

考虑通过分治优化DP

在solve(l,r)时,求出[l+1,r]中c[]最大的位置,设为k

以k为分界线可以递归solve(l,k-1),solve(k,r)

然后只需用[l,k-1]的决策更新[k,r]即可

由于c[k]最大,所以c[k]<=i-j

i从max(c[k]+l,k)开始,决策j一开始的取值范围为[max(l,g[i]),i-c[k]]

每当i往右移一位时,j的上限也往右移一位,可以做到$O(1)$更新

j的下限可能也会右移到g[i],此时有l<=g[i]<=k-1,由于所有更新i的区间[l,k-1]均不相交

所以只存在一个区间[l,k-1]满足l<=g[i]<=k-1,即每个i最多只会发生一次下限右移

对于每次右移用线段树查询新区间内的最优解即可

当i循环到k+c[k]时,[k+c[k],r]内所有i的可行决策j的上限都为k-1,所以按g[]值将

[k+c[k],r]分割,对于每一段用线段树区间更新即可

总体复杂度为$O(n\log n)$。

#include<cstdio>
const int N=1000010,M=2097153,P=1000000007,inf=-1000000;
int n,i,j,c[N],d[N],g[N];
inline void read(int&a){char c;while(!(((c=getchar())>='0')&&(c<='9')));a=c-'0';while(((c=getchar())>='0')&&(c<='9'))(a*=10)+=c-'0';}
inline int min(int a,int b){return a<b?a:b;}
inline int max(int a,int b){return a>b?a:b;}
inline int merge(int a,int b){return c[a]>c[b]?a:b;}
struct Num{
int x,y;
Num(){x=y=0;}
Num(int _x,int _y){x=_x,y=_y;}
inline Num operator+(Num b){
if(x<b.x)return b;
if(x>b.x)return Num(x,y);
return Num(x,(y+b.y)%P);
}
inline Num operator+(int _x){return Num(x+_x,y);}
inline void operator+=(Num b){*this=*this+b;}
}f[N];
struct Node{int c,d;Num f,tag;}T[M];
void build(int x,int a,int b){
if(a==b){
T[x].c=a;
T[x].d=d[a];
return;
}
int mid=(a+b)>>1;
build(x<<1,a,mid),build(x<<1|1,mid+1,b);
T[x].c=merge(T[x<<1].c,T[x<<1|1].c);
T[x].d=min(T[x<<1].d,T[x<<1|1].d);
}
void build2(int x,int a,int b){
T[x].tag=Num(inf,0);
if(a==b){
T[x].d=d[a];
T[x].f=f[a];
return;
}
int mid=(a+b)>>1;
build2(x<<1,a,mid),build2(x<<1|1,mid+1,b);
T[x].d=min(T[x<<1].d,T[x<<1|1].d);
T[x].f=T[x<<1].f+T[x<<1|1].f;
}
int askc(int x,int a,int b,int c,int d){
if(c<=a&&b<=d)return T[x].c;
int mid=(a+b)>>1;
if(d<=mid)return askc(x<<1,a,mid,c,d);
if(c>mid)return askc(x<<1|1,mid+1,b,c,d);
return merge(askc(x<<1,a,mid,c,d),askc(x<<1|1,mid+1,b,c,d));
}
int askd(int x,int a,int b,int c,int d){
if(c>d)return n+1;
if(c<=a&&b<=d)return T[x].d;
int mid=(a+b)>>1;
if(d<=mid)return askd(x<<1,a,mid,c,d);
if(c>mid)return askd(x<<1|1,mid+1,b,c,d);
return min(askd(x<<1,a,mid,c,d),askd(x<<1|1,mid+1,b,c,d));
}
void add(int x,int a,int b,int c,int d,Num p){
if(c<=a&&b<=d){T[x].tag=T[x].tag+p;return;}
int mid=(a+b)>>1;
if(c<=mid)add(x<<1,a,mid,c,d,p);
if(d>mid)add(x<<1|1,mid+1,b,c,d,p);
}
Num askf(int x,int a,int b,int c,int d){
if(c>d)return Num(inf,0);
if(c<=a&&b<=d)return T[x].f;
int mid=(a+b)>>1;
if(d<=mid)return askf(x<<1,a,mid,c,d);
if(c>mid)return askf(x<<1|1,mid+1,b,c,d);
return askf(x<<1,a,mid,c,d)+askf(x<<1|1,mid+1,b,c,d);
}
inline Num askf1(int c){
int x=1,a=0,b=n,mid;Num t=Num(inf,0);
while(a!=b){
t+=T[x].tag;
mid=(a+b)>>1,x<<=1;
if(c<=mid)b=mid;else a=mid+1,x|=1;
}
return t+T[x].tag;
}
void change(int x,int a,int b,int c,Num p){
if(a==b){T[x].f=p;return;}
int mid=(a+b)>>1;
if(c<=mid)change(x<<1,a,mid,c,p);else change(x<<1|1,mid+1,b,c,p);
T[x].f=T[x<<1].f+T[x<<1|1].f;
}
inline void update(int l,int k,int r){
int i=max(c[k]+l,k);
if(g[i]>=k||i>r)return;
int jl=max(l,g[i]),jr=i-c[k];
Num tmp=askf(1,0,n,jl,jr)+1;
for(;i<=k-1+c[k]&&i<=r;i++){
if(g[i]>jl){
if(g[i]>=k)return;
jl=g[i];
tmp=askf(1,0,n,jl,jr)+1;
}
f[i]+=tmp;
jr++;
if(jr>=jl)tmp+=f[jr]+1;
}
while(i<=r){
if(g[i]>jl){
if(g[i]>=k)return;
jl=g[i];
}
tmp=askf(1,0,n,jl,k-1)+1;
int t=askd(1,0,n,jl+1,n);
if(t>r){
add(1,0,n,i,r,tmp);
return;
}
add(1,0,n,i,t-1,tmp);
i=t;
}
}
void solve(int l,int r){
if(l==r){
if(l)change(1,0,n,l,f[l]=f[l]+askf1(l));
return;
}
int k=askc(1,0,n,l+1,r);
solve(l,k-1);
update(l,k,r);
solve(k,r);
}
int main(){
read(n);
for(i=1;i<=n;i++)read(c[i]),read(d[i]);
build(1,0,n);
for(i=0;i<=n;i++)d[i]=n+1,f[i]=Num(inf,0);
f[0]=Num(0,1);
for(i=0;i<=n;i++){
while(j<i&&i-j>askd(1,0,n,j+1,i))j++;
g[i]=j;
if(d[g[i]]>n)d[g[i]]=i;
}
build2(1,0,n);
solve(0,n);
if(f[n].x>0)printf("%d %d",f[n].x,f[n].y);else puts("NIE");
return 0;
}

  

BZOJ3711 : [PA2014]Druzyny的更多相关文章

  1. 【BZOJ3711】Druzyny

    [BZOJ3711]Druzyny 题面 bzoj 题解 首先我们有一个\(O(n^2)\)的\(dp\): 设\(f_i\)表示现在已经分好了\(1...i\)的组,且\(i\)作为一组的结尾的最大 ...

  2. P5979 [PA2014]Druzyny dp 分治 线段树 分类讨论 启发式合并

    LINK:Druzyny 这题研究了一下午 终于搞懂了. \(n^2\)的dp很容易得到. 考虑优化.又有大于的限制又有小于的限制这个非常难处理. 不过可以得到在限制人数上界的情况下能转移到的最远端点 ...

  3. [PA2014]Druzyny

    题目描述 体育课上,n个小朋友排成一行(从1到n编号),老师想把他们分成若干组,每一组都包含编号连续的一段小朋友,每个小朋友属于且仅属于一个组.第i个小朋友希望它所在的组的人数不多于d[i],不少于c ...

  4. @bzoj - 3711@ [PA2014]Druzyny

    目录 @description@ @solution@ @accepted code@ @details@ @description@ 体育课上,n个小朋友排成一行(从1到n编号),老师想把他们分成若 ...

  5. Work at DP

    转载请注明出处:http://www.cnblogs.com/TSHugh/p/8858805.html Prepared: (无notes的波兰题目的notes见我的波兰题目补全计划)BZOJ #3 ...

  6. bzoj AC倒序

    Search GO 说明:输入题号直接进入相应题目,如需搜索含数字的题目,请在关键词前加单引号 Problem ID Title Source AC Submit Y 1000 A+B Problem ...

  7. BZOJ3711 Druzyny 最大值分治、线段树

    传送门 被暴力包菜了,然而还不会卡-- 有一个很暴力的DP:设\(f_i\)表示给\(1\)到\(i\)分好组最多可以分多少组,转移枚举最后一个组.接下来考虑优化这个暴力. 考虑:对于每一个位置\(i ...

  8. BZOJ 3721: PA2014 Final Bazarek

    3721: PA2014 Final Bazarek Time Limit: 20 Sec  Memory Limit: 128 MBSubmit: 645  Solved: 261[Submit][ ...

  9. BZOJ 3709: [PA2014]Bohater

    3709: [PA2014]Bohater Time Limit: 5 Sec  Memory Limit: 128 MBSec  Special JudgeSubmit: 1050  Solved: ...

随机推荐

  1. 获取oracle 表字段,表名,以及主键之类等等的信息。

    获取表名:  Oracle的user_talbes用于记录了用户表信息. select * from user_tables  获取某个表的字段: USER_TAB_COLS中记录了用户表的列信息.下 ...

  2. TFS增加dataserver

    通过之前的努力,已经搭建好了一套基本的tfs环境,包括一台nameserver和一台dataserver以及独立的nginx-tfs,而在实际应用中的分布式文件系统,只有一台dataserver明显是 ...

  3. SQL小技巧小知识

    1.[ ]的使用 当我们所要查的表是系统关键字或者表名中含有空格时,需要用[]括起来,例如新建了两个表,分别为user,user info,那么select * from user和select * ...

  4. Sybase IQ如何将大文件数据迅速加载到数据库

    试想一下,如果一个文件5G.10G甚至更大.如何将它迅速地加载到数据库指定的表呢?我们看看Sybase IQ是如何迅速地将表的数据加载到数据库的. 数据文件格式: 1440,2011-01-09 00 ...

  5. DP:炮兵阵地问题(POJ 1185)

    正确的打炮方式(大雾)(点我查看) 2015-08-21 问题是中文的,大家可以进去看看. 先说一个坑,这个问题我交了很多次,都没过,反正是WA到我烦了,都不知道哪里错了!!!怎么会有错,然后翻了一下 ...

  6. js如何往数组Array中添加元素

    转载自:http://blog.sina.com.cn/s/blog_95fa28e60101mwup.html unshift:将参数添加到原数组开头,并返回数组的长度  pop:删除原数组最后一项 ...

  7. MFC基于Dialog的工程中使用OSG

    osg的例子有osgviewerMFC,是MDI类型的MFC工程,我一般用基于对话框的MFC较多. 注意观察MFC_OSG.h文件中的cOSG构造函数,参数是一个窗口句柄hWnd,这里的窗口可以不只局 ...

  8. Docker跨主机通信之路由

    一.实验环境: 主机名 主机IP Docker0_IP Docker1 192.168.88.130 172.17.0.1 Docker2 192.168.88.131 172.18.0.1 二.实验 ...

  9. Java中栈结构的自我实现

    package com.pinjia.shop.common.collection; /** * Created by wangwei on 2017/1/3. */ public class MyL ...

  10. Fresco 源码分析(三) Fresco服务端处理(3) DataSource到Producer的适配器逻辑以及BitmapMemoryCacheProducer处理的逻辑

    4.3.1.2.1 Producer和DataSource之间适配器处理的逻辑 还是从程序的入口开始说吧 CloseableProducerToDataSourceAdapter.create() 源 ...