原题链接

鸣谢:AGC 018E.Sightseeing Plan(组合 DP)

本蒟蒻认为,本题堪称网格路径问题观止。

因为涵盖了不少网格路径问题的处理方法和思路。

一句话题意:

给你三个矩形。

三个矩形从左下到右上排开。矩形顶点坐标范围是1e6

  • 1≤X1X2<X3X4<X5X6≤106
  • 1≤Y1Y2<Y3Y4<Y5Y6≤106

大概就是这样:

对于所有的1中选择一个点P1,2中选择一个点P2,3中选择一个点P3。

求从P1到P2再到P3的最短路径条数之和。

从一个矩形再到另外一个矩形还要经过一个矩形太复杂。我们化简情况处理下。

反正本质就是所有三个点之间的路径。

一、一个点到一个点

你必须要知道的:

从(x1,y1)到(x2,y2)的最短路径,就是只能往上或者往右走。

最短路径条数就是:C(x2-x1+y2-y1,x2-x1)

所以我们可以O(1000000^6)枚举

二、一个点到一个矩形

$F(x,y)$表示,从$(0,0)$到$(x,y)$的路径条数。

这个可以直接算。

发现,

$F(x,y)=\sum_{j=0}^yF(x-1,j)$

可以理解为,从原点到所有的$(x-1,j)$然后向上走一步,然后直接向右到达$(x,y)$

肯定不重不漏。

那么,可以得到:

从$(0,0)$到一个矩阵路径,就是:

$F(x2+1,y2+1)-F(x2+1,y1)-F(x1,y2+1)+F(x1,y1)$

就是一个小容斥。

用上面的$F(x,y)=\sum_{j=0}^yF(x-1,j)$

把$F(x2+1,y2+1)$还有$F(x1,y2+1)$展开消一消,

每次保留$F(x2+1,j)$和$F(x1,j)$往下消。

画画图就看出来了。

所以,我们得到了从一个点到一个矩阵的路径条数。

只要计算那四个点即可。

三、一个矩形到一个矩形

列出式子:

$G$示路径条数

$M_1$代表第一个矩形,$M_2$代表第二个矩形。

$\sum_{x1} \sum_{y1} \sum_{x2} \sum_{y2}G((x1,y1),(x2,y2))$

可以提出两个sigma

变成枚举一个点,求到另外一个矩形的方案数

$\sum_{x1} \sum_{y1}G((x1,y1),M_2)$

其实就是:

$\sum_{x1} \sum_{y1}\sum_{xx}\sum_{yy}(G((x1,y1),(xx,yy))$

$xx,yy$代表那四个关键点。(省略了四个关键点正负号)

反过来,每个关键点都会被$M_1$所有的点统计一次。

所以,一个关键点的贡献,就是这个关键点到$M_1$的路径条数。

就是这个关键点到$M_1$的四个关键点路径条数。(当然,要有正负号)

所以,一个矩形到一个矩形的路径条数,

就是两个矩形四个关键点分别进行条数计算。处理好符号即可。

四、一个矩形经过一个矩形再到另一个矩形

可以把三的结论推广。

就是,第一个矩形四个关键点,到第三个矩形四个关键点,然后路径上经过第二个矩形的方案数。

所以,$4\times 4$枚举第一第三个矩形的关键点的话,

问题就变成了从一个点出发,经过一个矩形再到另外一个点的方案数。

直接枚举还是$O(n^2)$的。

发现,经过第二个矩形,
必然要么从$(x,y3)$(下边界)要么从$(x3,y)$(左边界)
进入。

所以,我们枚举进入点。

进入这个点,就一定进入了这个矩形。再到终点的任意一条路径,都是合法的。

再乘上从起点到这个进入点的方案数就是这个进入点的贡献。

必然不重不漏地枚举完了所有的合法路径。

所以我们成功的A掉了这个题。

五、然鹅并没有做完。。。

发现,不是要求经过第二个矩形的路径条数啊,我们要在第二个矩形中选择一个点。。。。

一个经过第二个矩形的长度为$len$的路径,就对应着$len$个方案。(每个位置都可以作为$P_2$)

我们经常转化研究对象,尝试分开统计贡献。

分开统计的前提是,贡献必须可以处理成互不相关的形式。

这个可以不可以呢?

但是凉凉,这个$len$还和离开点有关。。。

难道只能$O(n^2)$枚举进入点和离开点?

不!我们还有方法!

一个$len$怎么计算?

进入点$(x1,y1)$,离开点$(x2,y2)$

有$len=x2-x1+y2-y1+1$

诶!!

$x1,x2,y1,y2$貌似可以分离!!

那么,我们可以枚举进入点。

贡献是:$(-1)\times$刚才的两个组合数$\times(x+y)$

离开点:

贡献是:$(+1)\times$刚才的两个组合数$\times(x+y)$

(其实这个也是不重不漏枚举了所有路径)

相加的话,

发现,对于每个合法的路径,恰好被计算了两遍。(进入离开点各一次)

第一遍把$-x1-y1$算上,

第二遍把$x2+y2+1$算上。

所以,每个合法路径贡献就是$len$次。和题意符合。

完结撒花!!!~~~

六、一些代码实现细节:

1.len的长度是:$(x2-x1+y2-y1+1)$别忘了$+1$

2.注意,矩阵矩阵的之间的路径,

根据推导,是先转化成$M_2$中关键点到$M_1$矩阵的路径,所以,$M_1$的矩阵的四个关键点分别是$(x1-1,y1-1),(x2,y1-1),(x1-1,y2),(x2,y2)$

类比之前,这个是从上往下走的,所以要这样处理。

#include<bits/stdc++.h>
#define ri register int
using namespace std;
typedef long long ll;
const int mod=1e9+;
const int N=2e6+;
ll ans=;
ll jie[N],ni[N];
struct node{
int x,y,fl;
void init(int a,int b,int c){
x=a,y=b,fl=c;
}
}p[];
int tot;
int x3,y3,x4,y4;
int xx[],yy[];
ll qm(ll x,ll y){
ll ret=;
while(y){
if(y&) ret=ret*x%mod;
x=x*x%mod;
y>>=;
}
return ret;
}
ll C(int n,int m){
return jie[n]*ni[m]%mod*ni[n-m]%mod;
}
ll G(int x1,int y1,int x2,int y2){
ll A=abs(x2-x1),B=abs(y2-y1);
return C(A+B,B);
}
ll sol(int x1,int y1,int x2,int y2,int f1,int f2){
ll ret=;
for(ri x=x3;x<=x4;++x){
ret=(ret+G(x1,y1,x,y4)*G(x,y4+,x2,y2)%mod*(x+y4+))%mod;//注意这里的x+y4+1的+1
ret=(ret-G(x1,y1,x,y3-)*G(x,y3,x2,y2)%mod*(x+y3)%mod+mod)%mod;
}
for(ri y=y3;y<=y4;++y){
ret=(ret+G(x1,y1,x4,y)*G(x4+,y,x2,y2)%mod*(x4+y+))%mod;
ret=(ret-G(x1,y1,x3-,y)*G(x3,y,x2,y2)%mod*(x3+y)%mod+mod)%mod;
}
ret=ret*(f1*f2); return ret;
}
int main(){
int x,y;
for(ri i=;i<=;++i)scanf("%d",&xx[i]);
for(ri i=;i<=;++i)scanf("%d",&yy[i]);
jie[]=;
for(ri i=;i<=N-;++i) jie[i]=jie[i-]*i%mod;
ni[N-]=qm(jie[N-],mod-);
for(ri i=N-;i>=;--i) ni[i]=ni[i+]*(i+)%mod; x3=xx[],y3=yy[],x4=xx[],y4=yy[];
p[].init(xx[]-,yy[]-,);p[].init(xx[],yy[]-,-);//注意这里是这样的
p[].init(xx[]-,yy[],-);p[].init(xx[],yy[],); p[].init(xx[],yy[],);p[].init(xx[]+,yy[],-);
p[].init(xx[],yy[]+,-);p[].init(xx[]+,yy[]+,); for(int i=;i<=;++i){
for(int j=;j<=;++j){
ans=(ans+sol(p[i].x,p[i].y,p[j].x,p[j].y,p[i].fl,p[j].fl)+mod)%mod;
}
}
printf("%lld",ans);
return ;
}

七、总结

我们成功地把一个O(n^6)变成了O(n)

思路就是化简问题,然后从简单到困难,利用之前的结论考虑,不断转化简化问题。

最后的差分len也是异常精彩。

组合数和路径条数的问题,经常是考虑一个物品的贡献。

可以尝试往这方面想。

这里也有有一个不错的路径组合意义转化题:fzyzojP3782 -组合数问题

AGC 018E.Sightseeing Plan——网格路径问题观止的更多相关文章

  1. AGC 018E.Sightseeing Plan(组合 DP)

    题目链接 \(Description\) 给定三个不相交的矩形\(A(X1,Y1)-(X2,Y2),B(X3,Y3)-(X4,Y4),C(X5,Y5)-(X6,Y6)\),求 从第一个矩形中某点\(a ...

  2. DP问题练习2:网格路径数量计算问题

    DP问题练习2:网格路径数量计算问题 问题描述 有一个机器人的位于一个 m × n 个网格左上角. 机器人每一时刻只能向下或者向右移动一步.机器人试图达到网格的右下角. 问有多少条不同的路径? 样例: ...

  3. [NOI2005]维护数列——平衡树观止

    本题题解并不详细,不推荐大家看这一篇. 可以看这篇 题目描述 请写一个程序,要求维护一个数列,支持以下 6 种操作:(请注意,格式栏 中的下划线‘ _ ’表示实际输入文件中的空格) 100%的数据中, ...

  4. 几何学观止(Lie群部分)

    上承这个页面,这次把Lie群的部分写完了 几何学观止-微分几何部分(20181102).pdf 我觉得其他部分(尤其是代数几何部分)我目前没有把握写得令自己满意,总之希望在毕业前能写完吧. 这次调整了 ...

  5. 几何学观止(Riemann流形部分)

    上承这个页面,相较之前,增加了古典的曲线曲面论,这部分介绍得很扼要,Riemann流形介绍得也很快,花了仅仅30页就介绍到了Gauss-Bonnet公式.同时配上了提示完整的习题. 几何学观止-Rie ...

  6. [AGC 018 E] Sightseeing plan

    STO ZKY ORZ Description 给定一张网格图和三个矩形,每次只能向上或向右走.你需要从矩形 \(A\) 中的一个点 \(S\) 出发,到达矩形 \(B\) 中的一个点 \(P\) , ...

  7. AtCoder Grand Contest 018 E Sightseeing Plan

    题意: 给定三个矩形,选定三个点,答案加上第一个点出发经过第二个点在第三个点结束的方案数,只能往右或往下走. 折腾了我半个多下午的题. 设三个矩形为$A,B,C$一个思路是枚举$B$的那个点$s(x, ...

  8. Python简介之探观止矣

    Python是一门什么样的编程语言编程语言主要分为编译型和解释型,静态语言和动态语言,强类型和弱类型,混合语言等.编译型语言:通过编译器把源代码编译(compile)成机器语言,在经过链接(linke ...

  9. CodeChef题目选讲

    https://wenku.baidu.com/view/2445a0322f60ddccda38a023.html 关键点:不超过7条 根据咕咕原理,所以答案最少是N/7;(N小于49就暴力) 随机 ...

随机推荐

  1. C# read write ini file

    [DllImport("kernel32")] private static extern long WritePrivateProfileString(string sectio ...

  2. WFP loading 窗口显示 SplashScreen

    public partial class App : Application { protected override void OnStartup(StartupEventArgs e) { Spl ...

  3. RabbitMQ基础教程之基本使用篇

    RabbitMQ基础教程之基本使用篇 最近因为工作原因使用到RabbitMQ,之前也接触过其他的mq消息中间件,从实际使用感觉来看,却不太一样,正好趁着周末,可以好好看一下RabbitMQ的相关知识点 ...

  4. Jenkins 构建运行java程序

    我们将在Jenkins建立执行一个简单的 HelloWorld 应用程序,构建和运行Java程序.打开网址:http://localhost:8080/jenkins 第1步- 转到Jenkins 仪 ...

  5. 深入理解docker信号机制以及dumb-init的使用

    一.前言 ● 容器中部署的时候往往都是直接运行二进制文件或命令,这样对于容器的作用更加直观,但是也会出现新的问题,比如子进程的资源回收.释放.托管等,处理不好,便会成为可怕的僵尸进程 ● 本文主要讨论 ...

  6. 机房ping监控 smokeping+prometheus+grafana

    一.前言 1.本监控方案主要由smokeping+promethues+grafana组成.smokeping主要数据采集,promethues作为数据存储,grafana数据展示 2.其实smoke ...

  7. A1020. Tree Traversals(25)

    这是一题二叉树遍历的典型题,告诉我们中序遍历和另外一种遍历序列,然后求任何一种遍历序列. 这题的核心: 建树 BFS #include<bits/stdc++.h> using names ...

  8. Go语言实现数据结构(一)单链表

    1.基本释义 2.结构体设计 3.基本方法设计 4.Main函数测试 1. 基本释义 线性表包含两种存储方法:顺序存储结构和链式存储结构,其中顺序表的缺点是不便插入与删除数据:接下来我们重点实现基于G ...

  9. python 爬虫学习之路

    2016-6-18 --今天实现了第一个用urllib2实现的爬虫程序. --过程中发现 req = urllib2.Request(url,headers = headers) 总是报错: 主要原因 ...

  10. 《Linux内核设计与实现》第四章学习笔记——进程调度

                                                                        <Linux内核设计与实现>第四章学习笔记——进程调 ...