博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
【Foreign】登山 [DP][数学]
阅读量:4967 次
发布时间:2019-06-12

本文共 3063 字,大约阅读时间需要 10 分钟。

登山

Time Limit: 10 Sec  Memory Limit: 256 MB

Description

  恶梦是一个登山爱好者,今天他来到了黄山
  俗话说的好,不走回头路。所以在黄山,你只能往前走,或者往上走。
  并且很显然的是,当你走到山脊的时候,你不能够往上走,你只能往前走一步再往上走。
  抽象一点而言就是,你可以把黄山视为一个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个点互不相同。

Output

  输出只有一个整数Ans,表示恶梦爬上山顶的路径数对10^9+7取模的值。

Sample Input

  5 2
  5 0
  1 1

Sample Output

  27

HINT

  对于100%的数据,保证N<=100000,C<=1000。
  保证对于(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
2 #include
3 #include
4 #include
5 #include
6 #include
7 #include
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 }
View Code

 

转载于:https://www.cnblogs.com/BearChild/p/7737408.html

你可能感兴趣的文章
zabbix短信网关调用问题总结
查看>>
130242014034-林伟领-实验一
查看>>
Insert excel data into DB
查看>>
复制和输入-编程中
查看>>
SQLSERVER 处理两个日期相减
查看>>
区间+状压 [Haoi2016]字符合并
查看>>
ubuntu重新加载nginx配置文件
查看>>
Forbidden You don't have permission to access / on this server.
查看>>
Windows server 2008 R2中安装MySQL !
查看>>
Intellij Idea新建web项目(转)
查看>>
raspberry 安装apache2,使其支持ssl ,并创建自签名证书
查看>>
Trie树:应用于统计和排序
查看>>
C语言结构体和函数
查看>>
PHP 删除目录及目录下文件
查看>>
[BZOJ3449] [Usaco2014 Feb]Secret Code
查看>>
XHTML与HTML区别
查看>>
软考-程序设计语言基础(编译原理)
查看>>
2016峰会:项目管理与高级项目管理(广州站)
查看>>
用JAVA编写浏览器内核之实现javascript的document对象与内置方法
查看>>
linux 命令之top
查看>>