【Foreign】登山 [DP][数学]
登山
Time Limit: 10 Sec Memory Limit: 256 MBDescription
恶梦是一个登山爱好者,今天他来到了黄山
俗话说的好,不走回头路。所以在黄山,你只能往前走,或者往上走。
并且很显然的是,当你走到山脊的时候,你不能够往上走,你只能往前走一步再往上走。
抽象一点而言就是,你可以把黄山视为一个N * N格点图,恶梦从(0,0)开始出发,要走到 (N,N)。
当他走到位置(x,y)的时候,它可以往(x + 1,y),或(x,y+1)走。
并且当他走到(x,x)的时候,由于他已经处在了山脊上,所以他不能够往(x,x+1)方向上走。
当恶梦兴致勃勃准备开始爬山的时候,他的同伴告诉他,黄山由于年久失修,有一些位置出现了大坑,不能走。
恶梦觉得更刺激了,但他想先知道他能有多少种方式走到黄山顶。
由于这个数字很大,所以你只需要将答案对10^9 + 7取模输出即可。
俗话说的好,不走回头路。所以在黄山,你只能往前走,或者往上走。
并且很显然的是,当你走到山脊的时候,你不能够往上走,你只能往前走一步再往上走。
抽象一点而言就是,你可以把黄山视为一个N * N格点图,恶梦从(0,0)开始出发,要走到 (N,N)。
当他走到位置(x,y)的时候,它可以往(x + 1,y),或(x,y+1)走。
并且当他走到(x,x)的时候,由于他已经处在了山脊上,所以他不能够往(x,x+1)方向上走。
当恶梦兴致勃勃准备开始爬山的时候,他的同伴告诉他,黄山由于年久失修,有一些位置出现了大坑,不能走。
恶梦觉得更刺激了,但他想先知道他能有多少种方式走到黄山顶。
由于这个数字很大,所以你只需要将答案对10^9 + 7取模输出即可。
Input
第一行包括两个整数N,C,分别表示你可以把黄山视作一个N * N的格点图,并且黄山上面有C个位置出现了大坑。
接下来的C行,每行包括两个整数X,Y,表示X,Y这个位置不能走。
保证X>=Y,也就是说(X,Y)必然在山上。
保证这C个点互不相同。
接下来的C行,每行包括两个整数X,Y,表示X,Y这个位置不能走。
保证X>=Y,也就是说(X,Y)必然在山上。
保证这C个点互不相同。
Output
输出只有一个整数Ans,表示恶梦爬上山顶的路径数对10^9+7取模的值。
Sample Input
5 2
5 0
1 1
5 0
1 1
Sample Output
27
HINT
对于100%的数据,保证N<=100000,C<=1000。
保证对于(0,0),(N,N)不存在障碍点。
保证对于(0,0),(N,N)不存在障碍点。
Solution
这显然是一道数学题,结合DP,我们令 f[i] 表示不经过其它障碍点,首先经过障碍点 i 的方案数。
那么显然有:f[i] = Ways(0,0 -> i) - f[j] * Ways(i -> j)。
问题就转化为了,怎样求出满足不超过直线y=x+1,从一点走向另外一点的方案数。
所以Ways = ((x1, y1) -> (x2, y2)) - ((x1, y1) -> (y2-1, x2+1))。
统计答案只要加入一个(n, n)在f里面计算即可。
Code
1 #include<iostream> 2 #include<string> 3 #include<algorithm> 4 #include<cstdio> 5 #include<cstring> 6 #include<cstdlib> 7 #include<cmath> 8 using namespace std; 9 typedef long long s64; 10 11 const int ONE = 5000005; 12 const int MOD = 1e9 + 7; 13 14 int n, m; 15 int x, y; 16 int fac[ONE], inv[ONE]; 17 int f[ONE]; 18 19 struct point 20 { 21 int x, y; 22 }a[ONE]; 23 bool cmp(const point &a, const point &b) 24 { 25 if(a.x != b.x) return a.x < b.x; 26 return a.y < b.y; 27 } 28 29 int get() 30 { 31 int res=1,Q=1; char c; 32 while( (c=getchar())<48 || c>57) 33 if(c=='-')Q=-1; 34 if(Q) res=c-48; 35 while((c=getchar())>=48 && c<=57) 36 res=res*10+c-48; 37 return res*Q; 38 } 39 40 int Quickpow(int a, int b) 41 { 42 int res = 1; 43 while(b) 44 { 45 if(b & 1) res = (s64)res * a % MOD; 46 a = (s64)a * a % MOD; 47 b >>= 1; 48 } 49 return res; 50 } 51 52 void Deal_first() 53 { 54 fac[0] = 1; 55 for(int i = 1; i <= 2 * n; i++) 56 fac[i] = (s64)fac[i - 1] * i % MOD; 57 inv[2 * n] = Quickpow(fac[2 * n], MOD - 2); 58 for(int i = 2 * n - 1; i >= 0; i--) 59 inv[i] = (s64)inv[i + 1] * (i + 1) % MOD; 60 } 61 62 int C(int n, int m) 63 { 64 if(n < 0 || m < 0) return 0; 65 return (s64)fac[n] * inv[m] % MOD * inv[n - m] % MOD; 66 } 67 68 void Modit(int &a) 69 { 70 if(a < 0) a += MOD; 71 if(a >= MOD) a -= MOD; 72 } 73 74 int Ways(point a, point b) 75 { 76 if(n < 0 || m < 0) return 0; 77 return C(b.y - a.y + b.x - a.x, b.y - a.y); 78 } 79 80 int Getit(point a, point b) 81 { 82 return Ways(a, b) - Ways(a, (point){b.y - 1, b.x + 1}); 83 } 84 85 int main() 86 { 87 n = get(); m = get(); 88 Deal_first(); 89 90 for(int i = 1; i <= m; i++) 91 a[i].x = get(), a[i].y = get(); 92 93 a[++m] = (point){n, n}; 94 sort(a + 1, a + m + 1, cmp); 95 96 for(int i = 1; i <= m; i++) 97 { 98 Modit(f[i] = Getit((point){0, 0}, a[i])); 99 for(int j = 1; j < i; j++) 100 Modit(f[i] -= (s64)f[j] * Getit(a[j], a[i]) % MOD); 101 } 102 103 printf("%d", f[m]); 104 }