zoukankan      html  css  js  c++  java
  • AtCoder Grand Contest 018 E

    题目传送门: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)=inom{x+y}{x})

    然后有个结论:(sumlimits_{i=0}^n W(i,m)=W(n,m+1)),证明只需要考虑我们枚举从第(m)列的哪个位置走到(m+1)

    我们记(G(x,y))表示从原点开始,到矩阵((0,0)-(x,y))的方案数,所以有

    (egin{align}G(x,y) &= sumlimits_{i=0}^xsumlimits_{j=0}^y W(i,j)\ &= sumlimits_{i=1}^{x+1} W(i,y)\ &=W(x+1,y+1)-1end{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_3leqslant xleqslant 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
    ",(Ans+p)%p);
    	return 0;
    }
    
  • 相关阅读:
    USACO6.4-The Primes
    ZOJ2112--Dynamic Rankings (动态区间第k大)
    Havel定理
    HDU5107---K-short Problem (线段树区间 合并、第k大)
    POJ2104-- K-th Number(主席树静态区间第k大)
    poj2409 & 2154 polya计数+欧拉函数优化
    CodeForces
    HDU
    HDU
    Gym
  • 原文地址:https://www.cnblogs.com/Wolfycz/p/10165651.html
Copyright © 2011-2022 走看看