zoukankan      html  css  js  c++  java
  • [51nod1486]大大走格子

      有一个h行w列的棋盘,里面有一些格子是不能走的,现在要求从左上角(1,1)走到右下角(h,w)的方案数。

     Input
      单组测试数据。
      第一行有三个整数h, w, n(1 ≤ h, w ≤ 10^5, 1 ≤ n ≤ 2000),表示棋盘的行和列,还有不能走的格子的数目。
      接下来n行描述格子,第i行有两个整数ri, ci (1 ≤ ri ≤ h, 1 ≤ ci ≤ w),表示格子所在的行和列。
      输入保证起点和终点不会有不能走的格子。
     Output
      输出答案对1000000007取余的结果。

      组合数学。设ans[i]表示从(1,1)走到第i个不能走的格子,且中间不经过其他障碍格的方案数。(把(h,w)也当成障碍格)

      ans[i]= C( X[i]-1+Y[i]-1 , X[i]-1 ) - sum{ ans[j]*C( X[i]-X[j]+Y[i]-Y[j] , X[i]-X[j] ) },点j在点i的左上方(包括左方或上方)。

     1 #include<cstdio>
     2 #include<iostream>
     3 #include<cstring>
     4 #include<algorithm>
     5 #include<queue>
     6 #define ll long long
     7 #define ull unsigned long long
     8 #define ui usigned int
     9 #define d double
    10 #define ld long double
    11 const int maxn=2023,modd=1000000007;
    12 struct zs{int x,y;}a[maxn];
    13 int jc[200233],ny[200233],ans[maxn];
    14 int i,j,k,n,m;
    15 
    16 int ra,fh;char rx;
    17 inline int read(){
    18     rx=getchar(),ra=0,fh=1;
    19     while(rx<'0'&&rx!='-')rx=getchar();
    20     if(rx=='-')fh=-1,rx=getchar();
    21     while(rx>='0')ra=ra*10+rx-48,rx=getchar();return ra*fh;
    22 }
    23 
    24 inline int getc(int n,int m){
    25     return 1ll*jc[n]*ny[n-m]%modd*ny[m]%modd;
    26 }
    27 inline int get(int x,int y){
    28     return getc(x+y,y);
    29 }
    30 inline int poi(int a,int b){
    31     int c=1;
    32     while(b){
    33         if(b&1)c=1ll*c*a%modd;
    34         a=1ll*a*a%modd,b>>=1;
    35     }return c;
    36 }
    37 bool operator <(zs a,zs b){return a.x<b.x||(a.x==b.x&&a.y<b.y);}
    38 inline void UPD(int &x){if(x<0)x+=modd;}
    39 int main(){
    40     n=read(),m=read();int num=read();register int i,j;
    41     for(i=jc[0]=ny[0]=1;i<=n+m;i++)jc[i]=1ll*jc[i-1]*i%modd,ny[i]=poi(jc[i],modd-2);
    42 //    for(i=1;i<=n||i<=m;i++)for(j=1;j<=i;j++)printf("%d %d  c:%d
    ",i,j,getc(i,j));
    43     for(i=1;i<=num;i++)a[i].x=read()-1,a[i].y=read()-1;
    44     a[++num]=(zs){n-1,m-1};
    45     std::sort(a+1,a+1+num);
    46     for(i=1;i<=num;i++){
    47         ans[i]=get(a[i].x,a[i].y);//printf("(%d,%d) %d
    ",a[i].x,a[i].y,ans[i]);
    48         for(j=1;j<i;j++)if(a[j].y<=a[i].y)
    49             UPD(ans[i]-=1ll*ans[j]*get(a[i].x-a[j].x,a[i].y-a[j].y)%modd);
    50     }
    51     printf("%d
    ",ans[num]);
    52 }
    View Code
  • 相关阅读:
    马拉车算法
    E. You 题解(思维)
    马拉车练习2
    The Boomsday Project 题解(玄学dp)
    Journey to Un'Goro 题解(思维+剪枝搜索)
    Black and white 题解(思维+prim)
    Rise in Price 题解(dp+随机数据)
    高斯消元
    马拉车练习1
    概率期望问题
  • 原文地址:https://www.cnblogs.com/czllgzmzl/p/5947491.html
Copyright © 2011-2022 走看看