[AGC009C]Division into 2
题意:
有一个长度为$N$的递增序列$S_i$,要把它分成$X,Y$两组,使得$X$中元素两两之差不小于$A$且$Y$中元素两两之差不小于$B$,求方案数
首先考虑$O\left(n^2\right)$的做法:
为了方便,我们令$S_0=-\infty$
设$f_{M,i,j}(M\in\{X,Y\},1\leq i\leq n,0\leq j\lt i)$表示已划分好$S_{1\cdots i}$且$S_j$是最后一个不属于$M$的元素的方案数
已算好$f_{X,1\cdots i,j}$和$f_{Y,1\cdots i,j}$,如何转移?
①若$S_{i+1}-S_i\geq A$,$S_{i+1}$可被放入$X$中,则$f_{X,i+1,0\cdots i-1}=f_{X,i,0\cdots i-1}$
否则$S_i,S_{i+1}$不可一起被放入$X$中,$f_{X,i+1,0\cdots i-1}=0$
②显然$f_{Y,i+1,i}=\sum\limits_{j=0}^{i-1}[S_{i+1}-S_j\geq B]f_{X,i,j}$
对$f_Y$的处理相似
最后的答案就是$\sum\limits_{i=0}^{n-1}f_{X,n,i}+\sum\limits_{i=0}^{n-1}f_{Y,n,i}$
#include<stdio.h>
#define ll long long
#define mod 1000000007
int fx[2010][2010],fy[2010][2010];
ll a[2010];
int main(){
int n,i,j;
ll A,B;
scanf("%d%lld%lld",&n,&A,&B);
for(i=1;i<=n;i++)scanf("%lld",a+i);
a[0]=-4223372036854775807ll;
fx[1][0]=fy[1][0]=1;
for(i=1;i<n;i++){
if(a[i+1]-a[i]>=A){
for(j=0;j<i;j++)fx[i+1][j]=fx[i][j];
}
if(a[i+1]-a[i]>=B){
for(j=0;j<i;j++)fy[i+1][j]=fy[i][j];
}
for(j=0;j<i;j++){
if(a[i+1]-a[j]>=B)fy[i+1][i]=(fy[i+1][i]+fx[i][j])%mod;
if(a[i+1]-a[j]>=A)fx[i+1][i]=(fx[i+1][i]+fy[i][j])%mod;
}
}
j=0;
for(i=0;i<n;i++)j=((j+fx[n][i])%mod+fy[n][i])%mod;
printf("%d",j);
}
考虑优化~
首先我们肯定不能开二维数组,考虑当前DP到$S_i$,只存$f_{M,j}$,并看一看当$i$变为$i+1$对答案的影响
因为$S$是递增的,所以满足$S_{i+1}-S_j\geq B$的$S_j$一定是一段前缀,所以我们可以用二分找到右端点并用线段树求区间和
其他转移就相当于线段树的单点更新
再用lazy tag实现清零即可
#include<stdio.h>
#define ll long long
#define mod 1000000007
int sumx[400010],sumy[400010],lazx[400010],lazy[400010],*laz,*sum,n;
ll a[100010];
void pushdown(int x){
if(laz[x]){
laz[x<<1]=laz[x<<1|1]=1;
sum[x<<1]=sum[x<<1|1]=0;
laz[x]=0;
}
}
int query(int L,int R,int l,int r,int x){
if(L<=l&&r<=R)return sum[x];
pushdown(x);
int mid=(l+r)>>1,ans=0;
if(L<=mid)ans=(ans+query(L,R,l,mid,x<<1))%mod;
if(mid<R)ans=(ans+query(L,R,mid+1,r,x<<1|1))%mod;
return ans;
}
void modify(int pos,int v,int l,int r,int x){
if(l==r){
sum[x]=(sum[x]+v)%mod;
return;
}
pushdown(x);
int mid=(l+r)>>1;
if(pos<=mid)
modify(pos,v,l,mid,x<<1);
else
modify(pos,v,mid+1,r,x<<1|1);
sum[x]=(sum[x<<1]+sum[x<<1|1])%mod;
}
int queryx(int L,int R){
laz=lazx;
sum=sumx;
return query(L,R,0,n-1,1);
}
void modifyx(int pos,int v){
laz=lazx;
sum=sumx;
modify(pos,v,0,n-1,1);
}
int queryy(int L,int R){
laz=lazy;
sum=sumy;
return query(L,R,0,n-1,1);
}
void modifyy(int pos,int v){
laz=lazy;
sum=sumy;
modify(pos,v,0,n-1,1);
}
int main(){
int i,l,r,mid,x,t1,t2;
ll A,B;
scanf("%d%lld%lld",&n,&A,&B);
for(i=1;i<=n;i++)scanf("%lld",a+i);
a[0]=-4223372036854775807ll;
modifyx(0,1);
modifyy(0,1);
for(i=1;i<n;i++){
l=0;
r=i-1;
while(l<=r){
mid=(l+r)>>1;
if(a[i+1]-a[mid]>=B){
x=mid;
l=mid+1;
}else
r=mid-1;
}
t1=queryx(0,x);
l=0;
r=i-1;
while(l<=r){
mid=(l+r)>>1;
if(a[i+1]-a[mid]>=A){
x=mid;
l=mid+1;
}else
r=mid-1;
}
t2=queryy(0,x);
if(a[i+1]-a[i]<A){
sumx[1]=0;
lazx[1]=1;
}
if(a[i+1]-a[i]<B){
sumy[1]=0;
lazy[1]=1;
}
modifyy(i,t1);
modifyx(i,t2);
}
printf("%d",(queryx(0,n-1)+queryy(0,n-1))%mod);
}
[AGC009C]Division into 2的更多相关文章
- AGC009C Division into Two
题意 有\(n\)个严格升序的数,请你分成两个集合\(A\)和\(B\),其中一个集合任意两数之差不小于\(x\),另一集合任意两数之差不小于\(y\). 问方案数,集合可以为空. $n \le 10 ...
- 【AGC009C】Division into Two
[AGC009C]Division into Two 题面 洛谷 题解 首先有一个比较显然的\(n^2\)算法: 设\(f_{i,j}\)表示\(A\)序列当前在第\(i\)个,\(B\)序列当前在第 ...
- python from __future__ import division
1.在python2 中导入未来的支持的语言特征中division(精确除法),即from __future__ import division ,当我们在程序中没有导入该特征时,"/&qu ...
- [LeetCode] Evaluate Division 求除法表达式的值
Equations are given in the format A / B = k, where A and B are variables represented as strings, and ...
- 关于分工的思考 (Thoughts on Division of Labor)
Did you ever have the feeling that adding people doesn't help in software development? Did you ever ...
- POJ 3140 Contestants Division 树形DP
Contestants Division Description In the new ACM-ICPC Regional Contest, a special monitoring and su ...
- 暴力枚举 UVA 725 Division
题目传送门 /* 暴力:对于每一个数都判断,是否数字全都使用过一遍 */ #include <cstdio> #include <iostream> #include < ...
- GDC2016【全境封锁(Tom Clancy's The Division)】对为何对应Eye Tracked System,以及各种优点的演讲报告
GDC2016[全境封锁(Tom Clancy's The Division)]对为何对应Eye Tracked System,以及各种优点的演讲报告 原文 4Gamer編集部:松本隆一 http:/ ...
- Leetcode: Evaluate Division
Equations are given in the format A / B = k, where A and B are variables represented as strings, and ...
随机推荐
- JAX-WS 注解
一.概述 “基于 XML 的 Web Service 的 Java API”(JAX-WS)通过使用注释来指定与 Web Service 实现相关联的元数据以及简化 Web Service 的开发.注 ...
- nginx 设置ip地址访问,但是设置域名访问不了
一.导语 在Nginx的设置过程中,ip地址能正常访问的,但是把ip地址转换成域名,就访问不了了,这个是怎么回事呢?今天来探讨一下 二.设置ip地址做负载均衡 2.1.server端 server { ...
- lucene、solr、nutch三者的关系
lucene是一个做搜索用的类库. nutch和solr都是基于lucene的,二者都是可直接运行的应用程序: 直接在业务上使用lucene的倒是不太多见. solr主要提供了建立索引(用户可以直接p ...
- HDU 5685 Problem A | 快速幂+逆元
Problem A Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/65536 K (Java/Others)Total S ...
- L2-001. 紧急救援---(Dijkstra,记录路径)
https://www.patest.cn/contests/gplt/L2-001 L2-001. 紧急救援 时间限制 200 ms 内存限制 65536 kB 代码长度限制 8000 B 判题程序 ...
- Python基础(6)_函数
一 为何要有函数? 不加区分地将所有功能的代码垒到一起,问题是: 代码可读性差 代码冗余 代码可扩展差 如何解决? 函数即工具,事先准备工具的过程是定义函数,拿来就用指的就是函数调用 结论:函数使用必 ...
- 【CodeForces】841D. Leha and another game about graph(Codeforces Round #429 (Div. 2))
[题意]给定n个点和m条无向边(有重边无自环),每个点有权值di=-1,0,1,要求仅保留一些边使得所有点i满足:di=-1或degree%2=di,输出任意方案. [算法]数学+搜索 [题解] 最关 ...
- #error,在xib文件中拷贝按钮所造成的错误.
https://www.evernote.com/shard/s227/sh/3e35a7b3-f40c-46df-8ae0-e7522310c18b/742311974127f12eaafae07a ...
- bzoj 1044 贪心二分+DP
原题传送门http://www.lydsy.com/JudgeOnline/problem.php?id=1044 首先对于第一问,我们可以轻易的用二分答案来搞定,对于每一个二分到的mid值 我们从l ...
- (十四)git操作
https://www.liaoxuefeng.com/wiki/0013739516305929606dd18361248578c67b8067c8c017b000