AtCoder Grand Contest 018 E - Sightseeing Plan
题目传送门:https://agc018.contest.atcoder.jp/tasks/agc018_e
题目大意:
一个人在网格图上旅行,他可以从矩形\((X_1,Y_1)-(X_2,Y_2)\)中任意一点\(S\)开始,再在矩形\((X_3,Y_3)-(X_4,Y_4)\)中任意一点\(P\)午休,最后在矩形\((X_5,Y_5)-(X_6,Y_6)\)中任意一点\(T\)结束旅行
如果旅行者当前在\((x,y)\),那么他下一步只能移动到\((x,y+1)\)或\((x+1,y)\)
只要点\(S,P,T\)不同或旅行经过的点集不同即为不同的方案,求有多少种不同的行走方案,答案对\(10^9+7\)取膜
我们记\(W(x,y)\)表示从原点到点\((x,y)\)的方案数,显然\(W(x,y)=\binom{x+y}{x}\)
然后有个结论:\(\sum\limits_{i=0}^n W(i,m)=W(n,m+1)\),证明只需要考虑我们枚举从第\(m\)列的哪个位置走到\(m+1\)列
我们记\(G(x,y)\)表示从原点开始,到矩阵\((0,0)-(x,y)\)的方案数,所以有
\(\begin{align}G(x,y) &= \sum\limits_{i=0}^x\sum\limits_{j=0}^y W(i,j)\\ &= \sum\limits_{i=1}^{x+1} W(i,y)\\ &=W(x+1,y+1)-1\end{align}\)
那么原点到矩阵\((x_1,y_1)-(x_2,y_2)\)的方案数\(f_{x_1,y_1,x_2,y_2}\)可以用容斥求得,即\(f_{x_1,y_1,x_2,y_2}=W(x_1,y_1)-W(x_2+1,y_1)-W(x_1,y_2+1)+W(x_2+1,y_2+1)\)
所以\(S\)所在矩阵可以拆成4个点:\((X_1-1,Y_1-1),(X_1-1,Y_2),(X_2,Y_1-1),(X_2,Y_2)\),系数分别为\(1,-1,-1,1\);同样的,\(T\)所在矩阵也可以拆成4个点(由于方向不同,因此稍有区别,具体见代码)
做完这样的处理后,可以发现题目变成了:给定起点\(S'\)和终点\(T'\),在矩阵\((X_3,Y_3)-(X_4,Y_4)\)中找一个休息点,求本质不同的路径数。可以发现,每条路径的贡献,就是其在矩形内部经过的点数,于是问题又转化为了求每条路径经过的点数和
这样我们依然可以利用容斥的思想,对于所有的\(X_3\leqslant x\leqslant X_4\),假定我们从\((x,Y_3-1)\)走到\((x,Y_3)\),那么我们就算出起点走到\((x,Y_3-1)\)的方案再从\((x,Y_3)\)走到终点的方案,这部分可以使用乘法原理,然后这条路径的贡献为\(-(x+Y_3)\)
对于其他的3条边,我们也按这种方法做(具体系数见代码),最后可以发现每条路径的系数恰好为其经过的点数
/*program from Wolfycz*/
#include<cmath>
#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
#define inf 0x7f7f7f7f
using namespace std;
typedef long long ll;
typedef unsigned int ui;
typedef unsigned long long ull;
inline char gc(){
static char buf[1000000],*p1=buf,*p2=buf;
return p1==p2&&(p2=(p1=buf)+fread(buf,1,1000000,stdin),p1==p2)?EOF:*p1++;
}
inline int frd(){
int x=0,f=1; char ch=gc();
for (;ch<'0'||ch>'9';ch=gc()) if (ch=='-') f=-1;
for (;ch>='0'&&ch<='9';ch=gc()) x=(x<<3)+(x<<1)+ch-'0';
return x*f;
}
inline int read(){
int x=0,f=1; char ch=getchar();
for (;ch<'0'||ch>'9';ch=getchar()) if (ch=='-') f=-1;
for (;ch>='0'&&ch<='9';ch=getchar()) x=(x<<3)+(x<<1)+ch-'0';
return x*f;
}
inline void print(int x){
if (x<0) putchar('-'),x=-x;
if (x>9) print(x/10);
putchar(x%10+'0');
}
const int N=2e6,p=1e9+7;
int fac[N+10],inv[N+10];
int X[10],Y[10],Ans;
void prepare(){
fac[0]=inv[0]=inv[1]=1;
for (int i=1;i<=N;i++) fac[i]=1ll*i*fac[i-1]%p;
for (int i=2;i<=N;i++) inv[i]=1ll*(p-p/i)*inv[p%i]%p;
for (int i=1;i<=N;i++) inv[i]=1ll*inv[i-1]*inv[i]%p;
}
struct S1{
int x,y,k;
void insert(int _x,int _y,int _k){x=_x,y=_y,k=_k;}
}A[4],B[4];
int F(int x1,int y1,int x2,int y2){return 1ll*fac[x2-x1+y2-y1]*inv[x2-x1]%p*inv[y2-y1]%p;}
int solve(const S1 &a,const S1 &b){
int x1=a.x,y1=a.y,x2=b.x,y2=b.y,res=0;
for (int i=X[3];i<=X[4];i++) res=(res+1ll*F(x1,y1,i,Y[3]-1)*F(i,Y[3],x2,y2)%p*(-i-Y[3])%p)%p;
for (int i=Y[3];i<=Y[4];i++) res=(res+1ll*F(x1,y1,X[3]-1,i)*F(X[3],i,x2,y2)%p*(-i-X[3])%p)%p;
for (int i=X[3];i<=X[4];i++) res=(res+1ll*F(x1,y1,i,Y[4])*F(i,Y[4]+1,x2,y2)%p*(i+Y[4]+1)%p)%p;
for (int i=Y[3];i<=Y[4];i++) res=(res+1ll*F(x1,y1,X[4],i)*F(X[4]+1,i,x2,y2)%p*(i+X[4]+1)%p)%p;
return res;
}
int main(){
prepare();
for (int i=1;i<=6;i++) X[i]=read();
for (int i=1;i<=6;i++) Y[i]=read();
A[0].insert(X[1]-1,Y[1]-1,1),A[1].insert(X[1]-1,Y[2],-1),A[2].insert(X[2],Y[1]-1,-1),A[3].insert(X[2],Y[2],1);
B[0].insert(X[6]+1,Y[6]+1,1),B[1].insert(X[6]+1,Y[5],-1),B[2].insert(X[5],Y[6]+1,-1),B[3].insert(X[5],Y[5],1);
for (int i=0;i<4;i++) for (int j=0;j<4;j++) Ans=(Ans+1ll*A[i].k*B[j].k*solve(A[i],B[j])%p)%p;
printf("%d\n",(Ans+p)%p);
return 0;
}