zoukankan      html  css  js  c++  java
  • 【bzoj2982】combination Lucas定理

    原文地址:http://www.cnblogs.com/GXZlegend/p/6801490.html


    题目描述

    LMZ有n个不同的基友,他每天晚上要选m个进行[河蟹],而且要求每天晚上的选择都不一样。那么LMZ能够持续多少个这样的夜晚呢?当然,LMZ的一年有10007天,所以他想知道答案mod 10007的值。(1<=m<=n<=200,000,000)

    输入

    第一行一个整数t,表示有t组数据。(t<=200)
    接下来t行每行两个整数n, m,如题意。

    输出

    T行,每行一个数,为C(n, m) mod 10007的答案。

    样例输入

    4
    5 1
    5 2
    7 3
    4 2

    样例输出

    5
    10
    35
    6


    题解

    Lucas定理:C(n , m) mod p = C(n/p , m/p) * C(n mod p , m mod p) mod p(p为质数)

    然后快速幂求个逆元就没啥了。

    #include <cstdio>
    #define mod 10007
    int fac[10010];
    int pow(int x , int y)
    {
    	int ans = 1;
    	while(y)
    	{
    		if(y & 1) ans = ans * x % mod;
    		x = x * x % mod , y >>= 1;
    	}
    	return ans;
    }
    int cal(int n , int m)
    {
    	if(n < m) return 0;
    	return fac[n] * pow(fac[m] , mod - 2) % mod * pow(fac[n - m] , mod - 2) % mod;
    }
    int lucas(int n , int m)
    {
    	if(!m) return 1;
    	return cal(n % mod , m % mod) * lucas(n / mod , m / mod) % mod;
    }
    int main()
    {
    	int t , n , m , i;
    	scanf("%d" , &t);
    	fac[0] = 1;
    	for(i = 1 ; i < mod ; i ++ ) fac[i] = fac[i - 1] * i % mod;
    	while(t -- )
    	{
    		scanf("%d%d" , &n , &m);
    		printf("%d
    " , lucas(n , m));
    	}
    	return 0;
    }

     

  • 相关阅读:
    解释之前遗留的方法覆盖问题
    多态在开发中的作用
    多态的基础语法
    Go 统计汉子字符
    Go map
    Go make和new的区别
    Go 指针
    Go 切片
    Go数组
    Go中交换两个值类型
  • 原文地址:https://www.cnblogs.com/GXZlegend/p/6801490.html
Copyright © 2011-2022 走看看