HDU5794 A Simple Chess 容斥+lucas

分析:转自http://blog.csdn.net/mengzhengnan/article/details/47031777

一点感想:其实这个题应该是可以想到的,但是赛场上并不会

dp[i]的定义很巧妙,容斥的思路也非常清晰

然后就是讨论lucas的用法,首先成立的条件是mod是素数

但是如果这个题mod很大,组合数取模感觉就不太可做了

我认为当mod很大时,n应该很小可以预处理,但是n很大时mod应该比较小,这样也可以预处理

如果n和mod都很大我就不会了。。。。

这个题在预处理的时候,同样需要预处理逆元,如果用费马小定理的话,比较慢是O(nlogmod)的

其实有更好的O(N)筛1到mod-1的算法

详情请参考:http://blog.miskcoo.com/2014/09/linear-find-all-invert

总的来说,赛场上没做出来还是太年轻

#include <algorithm>
#include <cstdio>
#include <vector>
#include <cstring>
using namespace std;
typedef long long LL;
const int N = 1e2+5;
const LL mod = 110119;
struct Node{
  LL x,y;
  bool operator<(const Node &rhs)const{
    return x<rhs.x;
  }
}p[N];
int r,kase;
LL n,m,dp[N],f[mod+5],inv[mod+5];
LL C(LL n, LL m){
    if(m > n) return 0;
    return f[n]*inv[f[m]]%mod*inv[f[n-m]]%mod;
}
LL lucas(LL n, LL m){
    if(m == 0) return 1;
    return C(n % mod, m % mod) * lucas(n / mod, m / mod) % mod;
}
int main(){
  f[1]=f[0]=inv[1]=1;
  for(int i=2;i<mod;++i){
    f[i]=f[i-1]*i%mod;
    inv[i]=inv[mod%i]*(mod-mod/i)%mod;
  }
  while(~scanf("%I64d%I64d%d",&n,&m,&r)){
    bool flag=false;
    for(int i=1;i<=r;++i){
      scanf("%I64d%I64d",&p[i].x,&p[i].y);
      if(p[i].x==n&&p[i].y==m)flag=true;
    }
    if(flag){
      printf("Case #%d: 0\n",++kase);
      continue;
    }
    sort(p+1,p+1+r);
    memset(dp,0,sizeof(dp));
    ++r;p[r].x=n,p[r].y=m;
    for(int i=1;i<=r;++i){
      LL tx=p[i].x-1,ty=p[i].y-1,a=-1,b=-1;
      if((tx+ty)%3||(2*tx-ty)%3)continue;
      a=(2*tx-ty)/3;if(a<0)continue;
      b=(tx+ty)/3-a;if(b<0)continue;
      dp[i]=lucas(a+b,a);
      for(int j=1;j<i;++j){
        if(!dp[j])continue;
        if(p[i].x==p[j].x||p[i].y<=p[j].y)continue;
        tx=p[i].x-p[j].x,ty=p[i].y-p[j].y,a=-1,b=-1;
        if((tx+ty)%3||(2*tx-ty)%3)continue;
        a=(2*tx-ty)/3;if(a<0)continue;
        b=(tx+ty)/3-a;if(b<0)continue;
        if(a==0&&b==0)continue;
        dp[i]=(dp[i]-dp[j]*lucas(a+b,a)%mod+mod)%mod;
      }
    }
    printf("Case #%d: %I64d\n",++kase,dp[r]);
  }
  return 0;
}

时间: 2024-10-01 13:32:45

HDU5794 A Simple Chess 容斥+lucas的相关文章

hdu5794 A Simple Chess 容斥+Lucas 从(1,1)开始出发,每一步从(x1,y1)到达(x2,y2)满足(x2?x1)^2+(y2?y1)^2=5, x2&gt;x1,y2&gt;y1; 其实就是走日字。而且是往(n,m)方向走的日字。还有r个障碍物,障碍物不可以到达。求(1,1)到(n,m)的路径条数。

A Simple Chess Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others)Total Submission(s): 2597    Accepted Submission(s): 691 Problem Description There is a n×m board, a chess want to go to the position (n,m) from the pos

hdu-5794 A Simple Chess(容斥+lucas+dp)

题目链接: A Simple Chess Time Limit: 2000/1000 MS (Java/Others)     Memory Limit: 65536/65536 K (Java/Others) Problem Description There is a n×m board, a chess want to go to the position (n,m) from the position (1,1).The chess is able to go to position (

A Simple Chess---hdu5794(容斥+Lucas)

题目链接:http://acm.split.hdu.edu.cn/showproblem.php?pid=5794 题意:给你一个n*m的网格,问从(1, 1)走到(n, m)的方案数是多少,其中有r个点是不可到达的: 根据公式我们可以知道每次只能走”日"型: 路径如上图所示,我们可以看到有很多点是不可达的,可达点都是满足(x+y)%3=2的;路径可以看成一个斜着放置的杨辉三角.我们只需要把坐标转换一下即可,这是没有障碍时的方案数: 让(1,1)到(n,m)中如果有一个障碍,那么我们可以用起点到

hdu_5794_A Simple Chess(lucas+dp)

题目链接:hdu_5794_A Simple Chess 题意: 给你n,m,从(1,1)到(n,m),每次只能从左上到右下走日字路线,有k(<=100)的不能走的位置,问你有多少方案 题解: 画图可看到路线是一个杨辉三角的图,然后我们可以将对应的x,y转换到对应的点上,也可以吧杨辉三角看成一个平行四边形, 我这里看成的平行四边形,设dp[i]为从起点到第i个障碍物的的方案数,那么dp[i]=dp[i]-sum(dp[j](第j个点能走到i这个点)*(j到i的方案数)). 然后我们把终点放到最后

HDU 5794 A Simple Chess(杨辉三角+容斥原理+Lucas)

题目链接 A Simple Chess 打表发现这其实是一个杨辉三角…… 然后发现很多格子上方案数都是0 对于那写可能可以到达的点(先不考虑障碍点),我们先叫做有效的点 对于那些障碍,如果不在有效点上,则自动忽略 障碍$(A, B)$如果有效,那么就要进行如下操作: 以这个点为一个新的杨辉三角的顶点,算出目标点的坐标$(x, y)$. 目标点的答案减去$C(A, B) * C(x, y)$的值. 但是这样会造成重复计算,原因是障碍之间可能有相互影响的关系. 这个时候就要考虑容斥原理,DFS消除这

HDU 5794 A Simple Chess(卢卡斯定理 + 容斥原理)

传送门 A Simple Chess Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others)Total Submission(s): 667    Accepted Submission(s): 168 Problem Description There is a n×m board, a chess want to go to the position (n,m) from the

cf451E Devu and Flowers 卢卡斯定理+容斥定理

题目:http://codeforces.com/problemset/problem/451/E 题意:有n个盒子(n<=20),每个盒子中有10^12个小球,现从每个盒子中取出若干球(可为0),求共取出s个小球(s<=10^14)的方案数. 组合数学问题,求C(n,m).但n,m过大时,可用卢卡斯定理. 卢卡斯定理:C(n,m) %p = C(n/p,m/p) * C(n%p,m%p) 从n个盒子中取出s个球的方案数,相当于插板,即 C(s+n-1,n-1).注意这是没有限制条件的情况.

HDU 6397(2018多校第8场1001) Character Encoding 容斥

听了杜教的直播后知道了怎么做,有两种方法,一种构造函数(现在太菜了,听不懂,以后再补),一种容斥原理. 知识补充1:若x1,x2,.....xn均大于等于0,则x1+x2+...+xn=k的方案数是C(k+m-1,m-1)种(貌似紫书上有,记不太清了). 知识补充2:若限制条件为n(即x1,x2....xn均小于n,假设有c个违反,则把k减掉c个n(相当于把c个超过n的数也变成大于等于0的),就可以套用知识1的公式了. 则最后的答案为sum( (-1)^c * C(m , c) * C(m-1+

Luogu4640 BJWC2008 王之财宝 容斥、卢卡斯定理

传送门 题意:有$N$种物品,其中$T$个物品有限定数量$B_i$,其他则没有限定.问从中取出不超过$M$个物品的方案数,对质数$P$取模.$N,M \leq 10^9 , T \leq 15 , P \leq 10^5$ 在$N$种物品中选出不超过$M$中物品的方案数可以用插板法(插板法只能满足刚好$M$个,那么我们可以虚构出一个数量无限的物品,把剩下的没选择完的都丢给它,这样插板法就能做了) 发现$T$很小,直接容斥即可 $N,M \leq 10^9$不能直接预处理,考虑到$P$为质数且范围