(lucas) Saving Beans

题目:

Although winter is far away, squirrels have to work day and night to save beans. They need plenty of food to get through those long cold days. After some time the squirrel family thinks that they have to solve a problem. They suppose that they will save beans in n different trees. However, since the food is not sufficient nowadays, they will get no more than m beans. They want to know that how many ways there are to save no more than m beans (they are the same) in n trees.

Now they turn to you for help, you should give them the answer. The result may be extremely huge; you should output the result modulo p, because squirrels can’t recognize large numbers.
Input
The first line contains one integer T, means the number of cases.

Then followed T lines, each line contains three integers n, m, p, means that squirrels will save no more than m same beans in n different trees, 1 <= n, m <= 1000000000, 1 < p < 100000 and p is guaranteed to be a prime.
Output
You should output the answer modulo p.
Sample Input
2
1 2 5
2 1 5
Sample Output
3
3

Hint
For sample 1, squirrels will put no more than 2 beans in one tree. Since trees are different, we can label them as 1, 2 … and so on.
The 3 ways are: put no beans, put 1 bean in tree 1 and put 2 beans in tree 1. For sample 2, the 3 ways are:
put no beans, put 1 bean in tree 1 and put 1 bean in tree 2.

分析与解答

1.先说一下lucas
参考:https://blog.csdn.net/wyg1997/article/details/52152282
https://blog.csdn.net/wyg1997/article/details/52152282
lucas定理:

(图片来自https://blog.csdn.net/arrowlll/article/details/53064748)
其实我们想一下二进制,111:就是1* 2^0+1*2^1+1*2^2
这个定理也是一样的akak-1…..a1a0就是p进制的每一位
所以有如下解释:

A、B是非负整数,p是质数。AB写成p进制:A=a[n]a[n-1]…a[0],B=b[n]b[n-1]…b[0] 这里的每一个数组元素表示其p进制的每一位。
则组合数C(A,B)=C(a[n],b[n])C(a[n-1],b[n-1])…*C(a[0],b[0])。也就是说,把大组合数问题变成了一个个的小组合数。(A,B小于mod)

其实可以想一下快速幂,里面%2就是取二进制的最后一位数,/2就是二进制向前移动一位,所以怎么实现代码也和快速幂类似,如下:

表达式:C(n,m)%p=C(n/p,m/p)*C(n%p,m%p)%p。(可以递归)
这里如果n/p和m/p很大,我们仍然可去调用lucas
递归方程:(C(n%p, m%p)*Lucas(n/p, m/p))%p。(递归出口为m==0,return 1)

LL Lucas(LL n,LL k)     //Lucas定理递归 
{if (k == 0)     //递归终止条件 return 1;elsereturn C(n % mod , k % mod) * Lucas(n / mod , k / mod) % mod;
}

对于每一个c(n,k)我们要写一个函数也就是求(n! / ( k! * ( n - k)! ) )%mod,由于有除法取模,所以我们要求k!*(n - k)!的逆元。
这里用小费马求逆元,我们还需写一个快速幂。由于组合数里面有阶乘,所以我们还需打一个阶乘表

整体下来就是这个样子
这里对mod取模

#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;
#define CLR(a,b) memset(a,b,sizeof(a))
#define INF 0x3f3f3f3f
#define LL long long
LL mod = 1000003;
LL fac[1000000+11] = {1,1};
void getfac()       //打一个阶乘表 
{for (int i = 2 ; i <= 1000000 ; i++)fac[i] = fac[i-1] * i % mod;
}
LL quick_mod(LL n , LL m)       //求快速幂 
{LL ans = 1;n %= mod;while (m){if (m & 1)ans = ans * n % mod;n = n * n % mod;m >>= 1;}return ans;
}
LL C(LL n , LL k)       //费马小定理求逆元 
{if (k > n)return 0;elsereturn fac[n] * (quick_mod(fac[k] * fac[n-k] % mod , mod - 2)) % mod;
}
LL Lucas(LL n,LL k)     //Lucas定理递归 
{if (k == 0)     //递归终止条件 return 1;elsereturn C(n % mod , k % mod) * Lucas(n / mod , k / mod) % mod;
}
int main()
{getfac();LL n,k;int Case = 1;int u;scanf ("%d",&u);while (u--){scanf ("%lld %lld",&n,&k);printf ("Case %d: %lld\n",Case++,Lucas(n,k));}return 0;
}

2.再说一下这个题
m个小球放到n个盒子,盒子可以为空, 方案数就是:C(n+m-1,n-1)
这个题是m个果子放到n个树上,类比一下,数就是盒子,小球就是果子,那么方案数就是C(n+m-1,n-1)
现在这个果子可以是0到m个,
所以总方案数
=C(n+m-1,n-1)+C(n+m-2,n-1)+…+C(n+1,n-1)+C(n,n-1)+C(n-1,n-1)
有一个组合数公式c(m,n)=c(m-1,n-1)+c(m-1,n)
这个组合数公式也可以看作:C(m,n-1)+C(m,n)=C(m+1,n)
还有个组合数公式:C(n,n)=1
所以我们从总方案数的后面变形
C(n-1,n-1)=C(n,n)
C(n-1,n-1)+C(n-1,n-1)=C(n,n)+C(n,n-1)=C(n+1,n)
现在后三个
C(n+1,n-1)+C(n,n-1)+C(n-1,n-1)=C(n+1,n-1)+C(n+1,n)=C(n+2,n)
我们发现,C(n+1,n-1)+C(n,n-1)+C(n-1,n-1)
这里m从2到0,上式等于C(n+2,n)
那么我们推广到整个式子
C(n+m-1,n-1)+C(n+m-2,n-1)+…+C(n+1,n-1)+C(n,n-1)+C(n-1,n-1)
这里m从m到0,式子等于C(n+m,n)
推到这我们现在就只需求
C(n+m,m) % p

(我觉得我应该去学数学。。我爱数学啊)
现在这个题p不是固定的每次都会变,所以我们不能用打表了
C(n,m)=n!/(m!(n-m)!)
=(n*(n-1) (n-m+1))/m!
分子从n-0一直到n-(m-1)也就是说循环m-1-0+1次也就是m次,而分母m!从1乘到m也是m次,那么我们就可以每次通过一个循环计算出C(n,m)了

ll C(ll n,ll m,ll p){if(m==0) return 1;if(m>n-m) m=n-m;ll up=1,down=1;for(int i=1;i<=m;i++){//这里不能打表了 up = (up*(n-i+1))%p;down=(down*i)%p; }return up*Pow(down,p-2,p)%p;
}

参考:
https://blog.csdn.net/qq_27717967/article/details/51493877
这个是wrong answer 我也不知道为什么

//当取模的p小于1e9时,不用预处理
#include <stdio.h>
#include <string.h>
#include <algorithm>
using namespace std;
typedef long long ll;
//计算n的k次方对M取模,二分法
ll Pow(ll n, ll k, ll p){ll ans=1;while(k){if(k&1)ans=(ans*n)%p;n = (n*n)%p;k>>=1; //k=k>>1  k=k/2;}return ans;
}
ll C(ll n,ll m,ll p){if(m==0) return 1;if(m>n-m) m=n-m;ll up=1,down=1;for(int i=1;i<=m;i++){//这里不能打表了 up = (up*(n-i+1))%p;down=(down*i)%p; }return up*Pow(down,p-2,p)%p;
}
ll lucas(ll n,ll m,ll p){if(m==0) return 1;return C(n%p,m%p,p)*lucas(n/p,m/p,p);
}
int main(){int t=1;ll m,n,p;scanf("%d",&t);while(t--){scanf("%I64d%I64d%I64d",&n,&m,&p);printf("%I64d\n",lucas(n+m,m,p));}return 0;
}

后来改成预处理,lucas也改变,才对。
但是这个代码改到上一个代码的那个题上,就runtime error
我也不知道为什么,玄学就玄学到这了

#include <iostream>
#include <string.h>
#include <cmath>
#define ll long long
using namespace std;
const int maxn=100002;
ll n,m,p;
ll fac[maxn];void getfac(ll p)//预处理阶层
{fac[0]=1;for(int i=1;i<=p;i++)fac[i]=fac[i-1]*i%p;
}ll power(ll a,ll n,ll p)//快速幂运算
{ll ans=1;while(n){if(n&1)ans=ans*a%p;a=a*a%p;n/=2;}return ans;
}ll lucas(ll n,ll m,ll p)
{ll ans=1;while(n&&m){ll a=n%p;ll b=m%p;if(a<b) return 0;ans=(ans*fac[a]*power(fac[b]*fac[a-b]%p,p-2,p))%p;//  fac[b]*fac[a-b]后面别忘了%p,否则WAn/=p;m/=p;}return ans;
}int main()
{int t;cin>>t;while(t--){cin>>n>>m>>p;getfac(p);cout<<lucas(n+m,m,p)<<endl;}return 0;
}

输入数据第一行是一个正整数T,表示数据组数 (T <= 100) 接下来是T组数据,每组数据有3个正整数 n, m, p (1 <= m <= n <= 10^9, m <= 10^4, m < p < 10^9, p是素数)第一个代码对

1 <= n, m <= 1000000000, 1 < p < 100000 and p is guaranteed to be a prime.第二个代码对

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/560670.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

java applet程序设计,Java Applet程序设计基础

Java Applet程序设计基础Java Applet 是用Java 语言编写的一些小应用程序&#xff0c;这些程序是直接嵌入到页面中&#xff0c;由支持Java的浏览器(IE 或 Nescape)解释执行能够产生特殊效果的程序。它可以大大提高Web页面的交互能力和动态执行能力。包含Applet的网页被称为Java…

(矩阵快速幂)解所有类似Fibonacci 的题目

Description In the Fibonacci integer sequence, F0 0, F1 1, and Fn Fn − 1 Fn − 2 for n ≥ 2. For example, the first ten terms of the Fibonacci sequence are: 0, 1, 1, 2, 3, 5, 8, 13, 21, 34, … An alternative formula for the Fibonacci sequence is Gi…

java ssm框架登录代码,求一个SSM框架登录功能的源码,要求能运行成功

[XML] 纯文本查看 复制代码<?xml version"1.0" encoding"UTF-8"?>xmlns:xsi"http://www.w3.org/2001/XMLSchema-instance"xmlns:aop"http://www.springframework.org/schema/aop"xmlns:context"http://www.springframewo…

php getdbused,PHP之购物车

该文章记录了购物车的实现代码&#xff0c;仅供参考book_sc_fns.phpinclude_once(output_fns.php);include_once(book_fns.php);include_once(db_fns.php);include_once("user_auth_fns.php");include_once("admin_fns.php");include_once("data_vali…

java中spring的web支持nio,Spring WebClient NIO功能和问题域,与Spring Webflux一起使用

我正在使用最新版本的Spring - Spring 5 .我正在开发http客户端的WebService“聚合器”&#xff0c;有点像路由请求到外部WebServices&#xff0c;接收响应后接收响应&#xff0c;做一些数据操作并回复我的HTTP服务的客户端 .为了在我的应用程序中创建http客户端&#xff0c;我…

2018.9.15,Arduino—流水灯实验报告

实验任务和目的 通过Arduino控制LED形成流水灯效果 实验条件 Arduino UNO&#xff0c;面包板&#xff0c;6个LED&#xff0c;6个220Ω电阻 实验过程和结果 实验详细步骤&#xff1a; 在各LED正极和Arduino引脚之间串联一个限流电阻&#xff0c;并将LED负极与Arduion的GND相连 …

php封装redis类,php封装redis操作类

Redis系列-key相关主要操作函数_数学_自然科学_专业资料。这篇 blog 主要总结下,redis 中跟 key 相关的常用函数 1)keys 语法:keys pattern 解释:查找所有匹配指定......Redis 实现分析网易杭研——胡炜 OUTLINE ? REDIS的内部实现(基于2.8.7)––––––––– 单线程模型的…

liunx php apache2,linux apache2部署php

android-plugmgr源代码分析android-plugmgr是一个Android插件加载框架,它最大的特点就是对插件不需要进行任何约束.关于这个类库的介绍见作者博客,市面上也有一些插件加载框架,但是感觉没有这个好.在这篇文章中,我 ...JS延时提示框p.p1 { margin: 0.0px 0.0px 0.0px 0.0px; fon…

2018.9.10.Matlab实验一:熟悉Matlab开发环境

一、实验任务和目的 1. 熟悉Matlab的系统环境及基本操作方法。 2. 掌握Matlab的搜索路径及其设置方法。 3. 熟悉Matlab的帮助信息的查阅方法。 二、实验内容 1. 熟悉 Matlab 工作界面的多个常用窗口的及使用方法。 熟悉Command windows、Workspace、Command history、C…

2018.9.10.Matlab实验二:Matlab基本运算

实验二&#xff1a;Matlab基本运算 一、实验任务和目的 1. 掌握变量的定义与数据类型。 2. 掌握变量的初始化方法。 3. 掌握数组、多维数组与子数组的定义、存储、赋值、变换。 4. 掌握逻辑数组的用法。 5. 熟悉MATLAB常用的函数、常用标点和快捷键。 二、实验内容 1. …

php进程数是指什么,25.查看php 某个服务的进程数

查看进程就是使用ps命令而已&#xff0c;只不顾ps的参数太多了。使用php查询的话&#xff0c;必须要开启几个函数(可以执行外部程序的函数)&#xff0c;参考官网&#xff1a;http://php.net/manual/zh/book.exec.php下面是在php进程中查询某个服务创建的进程数&#xff0c;比如…

2018.9.15,Matlab实验三:字符串、单元数组和结构体

一、实验任务和目的 掌握Matlab的字符串常用函数及其操作方法。掌握Matlab的结构体的基本操作方法。掌握Matlab的元胞数组的基本操作方法。 二、实验内容字符串数组Str[‘hopes, dreams, hold up, old up’]&#xff0c;查找’O’出现的次数和位置。现有三个字符串变量s1“i”…

c++ 不能分配给为0的数组_【嵌入式C】你有想过quot;数组下标quot;为何从0开始吗?...

1、聊一聊相信大家都有看过电影&#xff0c;今天所分享的是其经典背景音乐&#xff0c;或许音乐响起你又会想起那条单纯、善良的秋田犬&#xff01;今天跟大家聊聊一个有意思的话题&#xff0c;C中的数组下标为啥是从0开始?或者说为什么现在大部分的编程语言都会选择从0开始索…

oracle解析md5,Oracle中的MD5加密详解

一、技术点1、 DBMS_OBFUSCATION_TOOLKIT.MD5DBMS_OBFUSCATION_TOOLKIT.MD5是MD5编码的数据包函数&#xff0c;但偶在使用select DBMS_OBFUSCATION_TOOLKIT.MD5(input_string >abc) a from Dual时&#xff0c;却有错误提示&#xff0c;看来该函数只能直接在程序包中调用&…

2018.9.19.Matlab实验四:Matlab程序设计

一、实验任务和目的 熟悉程序设计思想。掌握伪代码的编写方法。掌握分支语句和循环结构的用法。 二、实验内容 输入一个百分制成绩&#xff0c;要求输出成绩等级A、B、C、D、E&#xff0c;其中90-100为A&#xff0c;80-89为B&#xff0c;70-79为C&#xff0c;60-69为D&#…

sata接口 图解 定义_SATA协议简介

SATA协议简介1、概述本文档主要介绍SATA的发展历程以及SATA相关协议&#xff0c;为后续SATA驱动框架分析做基础知识准备。2、SATA简介2.1 SATA发展历程2.1.1 PATA硬盘PATA硬盘叫做并行ATA硬盘(Parellel ATA)。为了限制其信号放大系统产生的高噪声&#xff0c;常采用高达5V的电压…

oracle 修改2个表,oracle学习笔记2:创建修改表

1.创建表CREATE TABLE ORDERINFO(ORDERID NUMBER(*, 0) NOT NULL, ORDERCODE VARCHAR2(20 BYTE) NOT NULL, USERID NUMBER(*, 0) NOT NULL, MOBILEPHONE VARCHAR2(20 BYTE) NOT NULL, ADDRESS VARCHAR2(128 BYTE) NOT NULL, PRODUCTNUMERIC NUMBER(*, 0) DEFAULT (0) NOT NULL,…

30岁软件测试转产品_SENSORO 产品总监回特入选 2019 年福布斯中国 30 岁以下精英榜...

日前&#xff0c;福布斯中国发布 2019 年度 30 岁以下精英榜&#xff08;30 Under 30&#xff09;&#xff0c;SENSORO&#xff08;北京升哲科技有限公司&#xff09;产品总监回特 成功入选 U30 精英榜企业科技版块。福布斯中国表示&#xff0c;今年技术领域人才辈出&#xff0…

Arduino实现数码管动态显示

之前一直以为公共端要么是解地要么是接电源&#xff08;3.3或5v&#xff09;&#xff0c;但是今天弄了半天我发现只要接口有电位差即可 没听课导致课堂作业没写出来&#xff0c;回来调了两个小时才完成 题目&#xff1a; 三位数码管从01亮到99&#xff0c;用arduino实现 分析…