BZOJ 3939 [Usaco2015 Feb]Cow Hopscotch ——线段树 CDQ分治
显然dp[i][j]=ps[i-1][j-1]-sigma(dp[k<i][l<j],a[i][j]=a[k][l])
考虑对于每一种颜色都开一颗区间线段树,但是空间不够。
所以我们可以动态开节点的权值线段树即可。
因为ij写反了调了30min。
然后发现空间的问题我们可以分治啊,按照纵坐标分治,然后处理左半边对右半边的影响即可。
然后CDQ分治即可,空间是O(nm)的,时间复杂度是O(nmlogm)的。
复杂度怎么算?主定理套用即可。
#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>
using namespace std;
#define F(i,j,k) for (int i=j;i<=k;++i)
#define D(i,j,k) for (int i=j;i>=k;--i)
#define md 1000000007
int sum[6000001],ls[6000001],rs[6000001],tot=0;
struct Dynamic_Segment_Tree{
int L,R,rt,X,C;
void init(){rt=0;}
int query(int x,int l,int r)
{
if (l>r) return 0;
if (!x) return 0;
int mid=l+r>>1;
if (L<=l&&r<=R) return sum[x];
if (R<=mid) return query(ls[x],l,mid);
else if (L>mid) return query(rs[x],mid+1,r);
else return (query(ls[x],l,mid)+query(rs[x],mid+1,r))%md;
}
void update(int x)
{
sum[x]=(sum[ls[x]]+sum[rs[x]])%md;
}
void modify(int &x,int l,int r)
{
if (!x) x=++tot;
int mid=l+r>>1;
if (l==r)
{
(sum[x]+=C)%=md;
return ;
}
if (X<=mid) modify(ls[x],l,mid);
else modify(rs[x],mid+1,r);
update(x);
}
}T[562501];
int dp[751][751],prs[751][751],r,c,k,a[751][751];
int main()
{
scanf("%d%d%d",&r,&c,&k);
F(i,1,r) F(j,1,c) scanf("%d",&a[i][j]);
F(i,1,k) T[i].init();
dp[1][1]=1;
F(i,1,r)
{
F(j,1,c)
{
(dp[i][j]+=prs[i-1][j-1])%=md;
T[a[i][j]].L=1;T[a[i][j]].R=j-1;
if (j>=2) (dp[i][j]=dp[i][j]-T[a[i][j]].query(T[a[i][j]].rt,1,c)+md)%=md;
prs[i][j]=(((prs[i][j-1]+prs[i-1][j])%md+dp[i][j])%md-prs[i-1][j-1]+md)%md;
}
F(j,1,c)
{
T[a[i][j]].X=j;T[a[i][j]].C=dp[i][j];
T[a[i][j]].modify(T[a[i][j]].rt,1,c);
}
}
printf("%d\n",dp[r][c]);
}
CDQ分治
#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>
using namespace std;
#define F(i,j,k) for (int i=j;i<=k;++i)
#define D(i,j,k) for (int i=j;i>=k;--i)
#define md 1000000007
int a[751][751],n,m,k,sum[1000005],dp[751][751],ps[751];
void CDQ(int l,int r)
{
if (l==r) return;
int mid=l+r>>1;
CDQ(l,mid);
F(i,1,n-1)
{
F(j,l,mid)
{
(sum[a[i][j]]+=dp[i][j])%=md;
(ps[i]+=dp[i][j])%=md;
}
(ps[i]+=ps[i-1])%=md;
F(j,mid+1,r)
{
(dp[i+1][j]+=ps[i])%=md;
dp[i+1][j]=(dp[i+1][j]-sum[a[i+1][j]]+md)%md;
}
}
F(i,1,n)
{
ps[i]=0;
F(j,l,mid) sum[a[i][j]]=0;
}
CDQ(mid+1,r);
}
int main()
{
scanf("%d%d%d",&n,&m,&k);
F(i,1,n) F(j,1,m) scanf("%d",&a[i][j]);
dp[1][1]=1;
CDQ(1,n);
printf("%d\n",dp[n][m]);
}
BZOJ 3939 [Usaco2015 Feb]Cow Hopscotch ——线段树 CDQ分治的更多相关文章
- 【BZOJ3939】[Usaco2015 Feb]Cow Hopscotch 动态规划+线段树
[BZOJ3939][Usaco2015 Feb]Cow Hopscotch Description Just like humans enjoy playing the game of Hopsco ...
- 【bzoj3939】[Usaco2015 Feb]Cow Hopscotch 动态开点线段树优化dp
题目描述 Just like humans enjoy playing the game of Hopscotch, Farmer John's cows have invented a varian ...
- [BZOJ 3888] [Usaco2015 Jan] Stampede 【线段树】
题目链接:BZOJ - 3888 题目分析 首先,计算出每个线段在 x 坐标 0 处出现的时间开始点和结束点,就转成了时间轴上的线段. 然后就是看每条线段是否被 y 比它小的线段完全覆盖了.注意求出的 ...
- BZOJ 1593: [Usaco2008 Feb]Hotel 旅馆 [线段树]
传送门 题意: 操作1:找长为$len$的空区间并填满,没有输出$0$ 操作2:将$[l,r]$之间的区间置空 我真是太弱了这种线段树还写了一个半小时,中间为了查错手动模拟了$30min$线段树操作, ...
- BZOJ 4025: 二分图 [线段树CDQ分治 并查集]
4025: 二分图 题意:加入边,删除边,查询当前图是否为二分图 本来想练lct,然后发现了线段树分治的做法,感觉好厉害. lct做法的核心就是维护删除时间的最大生成树 首先口胡一个分块做法,和hno ...
- COGS 577 蝗灾 线段树+CDQ分治
第一次写cdq分治 感谢hhd<y 这20亿对CP的指导(逃) 其实 就是 递归看左半部分对右半部分的贡献 (树状数组写挂了--临时改的线段树[大写的尴尬]) //By SiriusRen ...
- BZOJ3939 : [Usaco2015 Feb]Cow Hopscotch
设f[i][j]表示到(i,j)的方案数,则有 $f[i][j]=\sum f[x][y](x<i,y<j,a[x][y]!=a[i][j])=\sum f[x][y](x<i,y& ...
- BZOJ 4411: [Usaco2016 Feb]Load balancing 线段树+二分
code: #include <bits/stdc++.h> #define N 100060 #define M 1000000 #define lson x<<1 #def ...
- 【BZOJ 2957】楼房重建&&Codechef COT5 Count on a Treap&&【NOIP模拟赛】Weed 线段树的分治维护
线段树是一种作用于静态区间上的数据结构,可以高效查询连续区间和单点,类似于一种静态的分治.他最迷人的地方在于“lazy标记”,对于lazy标记一般随我们从父区间进入子区间而下传,最终给到叶子节点,但还 ...
随机推荐
- RAM建模和初始化
冯诺依曼提出的存储计算,计算存储,因此,几乎所有的CPU和ASIC都会使用存储器,它们的类型很多,包括异步RAM.同步RAM.ZBT RAM.DDR DRAM.ROM等.由于大部分的异步RAM和SRA ...
- Selenium私房菜系列5 -- 第一个Selenium RC测试案例
<Selenium简介>中讲过,Selenium RC支持多种语言编写测试案例,如:C#,Python.在工作中,我倾向于是用Python这类动态语言编写测试案例,因为这样的测试案例无需编 ...
- 在使用线程池时应特别注意对ThreadLocal的使用
使用ThreadLocal并且有线程池时要特别注意,ThreadLocal是以线程为key的,而线程池里面的线程是会被重新利用的,所以如果有使用线程池并且使用ThreadLocal来保存状态信息时要特 ...
- (九)maven之聚合多模块
聚合项目 一些开源项目,都会把自己的源代码公开到github之类的网站上,我们通过下载其代码,在本地执行maven install,可以把代码编译成jar包安装到本地仓库.而一个项目通常有多个模块,比 ...
- ubuntu 16.0 利用ant编译 hadoop-eclipse-plugins2.6.0
折腾了两天,抱着不放弃的精神,我终于编译出我自己所需的hadoop中在eclipse中的插件 在网上下载的可能因为版本不一致,在编译的时候出现各种各样的问题,包括你的eclipse版本和hadoop版 ...
- Luogu P5327 [ZJOI2019]语言
ZJOI2019Day2的温暖题,然后考场上只会大常数的\(O(n\log^3 n)\),就懒得写拿了60pts走人 首先我们简化题意,容易发现每个点能到达的点形成了一个联通块,我们只需要统计出这个联 ...
- Javascript 日期格式化
Javascript 日期格式化 需求: 给出:日期 .格式,根据日期格式进行输出. Date.prototype.Format = function (fmt) { //author: meizz ...
- 树形dp——Tree2cycle
一.问题描述(题目链接) 给你一棵树,删除或添加一条边的费用都是1,问使它变成一个环的最小费用. 二.解题思路 回溯法,然后回溯的时候的当前节点度数>2(如果是成环的话肯定就是2或者小于2)就把 ...
- python之for (循环)
格式: for 循环 for i in s: print(i) # for 关键字 # i 变量 # in 关键字 # s 可迭代对象 int - bool pass和- # for a in &qu ...
- 【求助】NdisSend,自定义数据包发送失败?
做ndis hook的时候,自定义了一个数据包,包结构应该没有问题,填充NDIS_PACKET结构是这样的,先初始化: NdisAllocatePacketPool(&nStat ...