数论4 组合数

目录

前言

求法一

代码

求法二

代码

求法三

代码

求法四

代码


前言

今天要将最后一部分,主要涉及组合数的四种求法。

前置知识

组合数的通项公式:
C_n^m = n! / {m!(n-m)!}

组合数的递推公式:

C_n^m = C_{n-1}^m + C_{n - 1} ^{m - 1}

卢卡斯定理:
C_n^m = C_{n\%p}^{m\%p} * C_{n/p}^{m/p}

我们今天需要求的四种求法主要基于这几个公式。


求法一

求法一利用的是递推公式,主要用于n <= 2000即可以打n^2的表的题目。

这个递推公式是怎样求出来的呢?其实很简单,我们单独对一个元素做分类讨论,显然有两种可能的情况:

  • 选择:C_{n - 1}^{m - 1}

  • 不选择:C_{n - 1} ^ {m}

最后二者相加能够得到的就是C_n^m,时间复杂度是O(n^2)


代码

const int N = 2010;
int C[N][N];
​
void init()
{for (int i = 0; i < N; i++){C[i][0] = 1; for (int j = 1; j <= i; j++)C[i][j] = (C[i - 1][j - 1] + C[i - 1][j]) % P;}
}

求法二

求法二主要应对的是n <= 1e5这种只能够打一维表的数据。我们使用通项公式求解。不过需要取模,而对于模意义下显然不可以直接做除法,需要求逆元

注意这里有个隐藏条件是P是质数,这样才能保证每一项都存在逆元。时间复杂度为O(nlogn)


代码

const int N = 100010, P = 10007;
int F[N], Fe[N];
​
int quick(int n, int k)
{int cnt = 1;while (k){if (k & 1) cnt = cnt * n % P;n = n * n % P;k >>= 1;}return cnt;
}
​
void init()
{F[0] = Fe[0] = 1;for (int i = 1; i < N; i++){F[i] = F[i - 1] * i % P;Fe[i] = Fe[i - 1] * quick(i, P - 2) % P;}
}

求法三

可算是找到了一道模板题(

这道题可以发现n很大,若再使用阶乘求得话时间复杂度就是nlogn是必定超时的。

对于这种情况我们就可以使用卢卡斯定理求解,当然依旧有个前提是P是质数

至于这个卢卡斯定理怎么求出来的我也不太懂,大家记住就好了,很形象,不难记。

若使用卢卡斯定理的话时间复杂度是:
log_p^n * p * log^p


代码

#include<iostream>
using namespace std;
const int P = 10007;
int t, m, n;
​
int quick(int n, int k)
{int cnt = 1;while(k){if(k & 1) cnt = cnt * n % P;k >>= 1;n = n * n % P;}return cnt;
}
​
int C(int n, int m)
{int l = 1;for(int i = 1, j = n; i <= m; i++, j--){l = l * j % P;l = l * quick(i, P - 2) % P; //乘以逆元}return l;
}
​
int lks(int n, int m)
{if(n < P) return C(n, m);return C(n % P, m % P) * lks(n/P, m/P) % P;
}
​
int main()
{scanf("%d", &t);while(t--){scanf("%d%d", &n, &m);printf("%d\n", lks(n, m));}}

求法四

n很大并且不取余

显然对于这样的问题就需要使用高精度来求解。

我们使用的依旧是通项公式

不过对于通项公式:C_n^m = n! / {m!(n-m)!}

里面是有乘法有除法的,这并不理想,有没有更优秀的求法呢?

这个求法很巧妙的,原理就是将阶乘分解质因数,然后消掉重复的部分,因为组合数都是整数所以上面部分是一定可以被下面部分整除的,所以大家不需要考虑消不完的情况。

那么我们如何对阶乘分解质因数呢?这个很简单,我们首先筛出所1~n中所有的质数,随后运用一个公式:
c = n/p + n/p^2 + n/p^3 + ...

对于这个公式主播最开始感觉有些摸不着头脑,但是想明白后只想说:妙,太妙了……

如何来理解呢?其实很简单n/p其实就是求出1 ~ n中能p整除的数字的个数,以此类推……


代码

#include<iostream>
#include<vector>
//#define int long long
using namespace std;
//typedef long long LL;
const int N = 1e6 + 10;
bool is_prime[N];
int n, m;
​
int quick(int n, int k)
{int l = 1;while (k){if (k & 1) l *= n;n = n * n;k >>= 1;}//printf("%d\n", l);return l;
}
​
vector<int> prime_shai(int n) //线性筛法
{vector<int> p;for (int i = 2; i <= n; i++){if (!is_prime[i]) p.push_back(i);for (int j = 0; p[j] <= n / i; j++){is_prime[i * p[j]] = true;if (i % p[j] == 0) break;}}return p;
}
​
int get_num(int n, int p)
{int l = 0;int cnt = p;while (n / p){l += n / p;p *= cnt;}return l;
}
​
vector<int> cur(vector<int>& A, int b)
{int x = 0;vector<int> C;for (int i = 0; i < A.size(); i++){x += A[i] * b;C.push_back(x % 10);x /= 10;}while (x){C.push_back(x % 10);x /= 10;}return C;
}
​
int main()
{scanf("%d%d", &n, &m);vector<int> A = { 1 }; //高精度vector<int> p = prime_shai(n);for (int i = 0; i < p.size(); i++){int l = get_num(n, p[i]) - get_num(m, p[i]) - get_num(n - m, p[i]);A = cur(A, quick(p[i], l));}for (int i = A.size() - 1; i >= 0; i--)printf("%d", A[i]);return 0;
}

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

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

相关文章

构建自己的私有 Git 服务器:基于 Gitea 的轻量化部署实战指南

对于个人开发者、小型团队乃至企业来说&#xff0c;将项目代码托管在 GitHub、Gitee 等公共平台虽然方便&#xff0c;但也存在一定的隐私与可控性问题。 搭建一套私有 Git 代码仓库系统&#xff0c;可以实现对源码的完全控制&#xff0c;同时不依赖任何第三方平台&#xff0c;…

Linux操作系统 4.Linux实用操作

一、各类小技巧&#xff08;快捷键&#xff09; 1.CTRL C 强制停止 1.Linux某些程序的运行&#xff0c;如果想要强行停止它&#xff0c;可以使用ctrlc 2.命令输入错误&#xff0c;也可以通过快捷键ctrl c,退出当前输入&#xff0c;重新输入&#xff0c;或者ctrlc跳过当前这…

react redux的学习,单个reducer

redux系列文章目录 一 什么redux&#xff1f; redux是一个专门用于做状态管理的JS库(不是react插件库)。它可以用在react, angular, vue等项目中, 但基本与react配合使用。集中式管理react应用中多个组件共享的状 简单来说&#xff0c;就是存储页面的状态值的一个库&#xf…

PCI与PCIe接口的通信架构是主从模式吗?

PCI&#xff08;Peripheral Component Interconnect&#xff09;总线在通信架构上本质是主从模式&#xff0c;但其具体实现和角色分配在不同版本&#xff08;如传统PCI与PCI Express&#xff09;中存在差异。以下是详细分析&#xff1a; 传统PCI总线的主从模式 (1) 基本架构 主…

java项目挂机自动重启操作指南

前段时间有个伙伴问我&#xff0c;java项目挂机怎么自动重启。。。。。。今天就写一个 .sh脚本来实现应用挂机的自动重启功能 #!/bin/bash # 查询mita的进程个数 countps -ef | grep mita.jar | grep -v "grep" | wc -l # echo $count nowtimedate "%Y-%m-%d %H…

开放最短路径优先 - OSPF【LSA详细】

目录 LSA的头部结构 LSA类型 LSA数据包 LSA的主要作用是传递路由信息。 LSA的头部结构 共占20个字节&#xff0c;不同类型的LSA头部字段部分都是相同的。 链路状态老化时间(Link-State Age) 2个字节。指示该条LSA的老化时间&#xff0c;即它存在了多长时间&#xff0c;单位…

SpringBoot+Spring+MyBatis相关知识点

目录 一、相关概念 1.spring框架 2.springcloud 3.SpringBoot项目 4.注解 5.SpringBoot的文件结构 6.启动类原理 二、相关操作 1.Jar方式打包 2.自定义返回的业务状态码 3.Jackson 4.加载配置文件 5.异常处理 三、优化配置 1.简化sql语句 2.查询操作 复杂查询 一…

《双影奇境》手机版上线?ToDesk用跨平台技术实现「全设备云电脑3A游戏」

《双影奇境》是由Hazelight Studios研发发行的一款双人合作冒险类游戏&#xff0c;玩家们在游戏中将扮演米欧和佐伊两位风格迥异的女作家&#xff0c;剧情讲述的是她们被骗进入一台意在窃取创意的机器后便陷入了自己创作的故事之中&#xff0c;并且必须相互依靠&#xff0c;努力…

【教程】Windows下 Xshell 连接跳板机和开发机

需求 使用远程连接工具 Xshell 连接跳板机&#xff0c;再从跳板机连接开发机&#xff0c;用户登陆方式为使用密钥。 方法 首先&#xff0c;建立一个会话&#xff0c;用于配置跳板机信息和开发机转跳信息&#xff1a; 在【连接】页面&#xff0c;给跳板机取个名字&#xff0c…

如何快速入门物联网单片机开发?

背景 物联网单片机硬件开发涉及多个阶段&#xff0c;元器件是否“自己设计”取决于具体需求。以下是详细解答和学习方案&#xff1a; 一、元器件是否自己设计&#xff1f; 通用元器件&#xff1a; 大多数情况下&#xff0c;开发者直接使用现成的标准化元器件&#xff08;如电阻…

每日一题(小白)模拟娱乐篇11

由题可知就是要求计算一个数字&#xff0c;可以整除10进制的每一位&#xff0c;亦可以整除8进制和16进制的每一位。要求找出第2023个能够在三个进制下同时被10进制整除的数字。 Java中已经封装了进制转换的方法&#xff0c;以下是一些常用的转换方法&#xff1a;&#x1f447;…

阿里巴巴langengine二次开发大模型平台

阿里巴巴LangEngine开源了&#xff01;支撑亿级网关规模的高可用Java原生AI应用开发框架 - Leepy - 博客园 阿里国际AI应用搭建平台建设之路(上) - 框架篇 基于java二次开发 目前Spring ai、spring ai alibaba 都是java版本的二次基础能力 重要的是前端工作流 如何与 服务端的…

MINIQMT学习课程Day8

获取qmt账号的资金账号后&#xff0c;我们进入下一步&#xff0c;如何获得当前账号的持仓情况 还是之前的步骤&#xff0c;打开qmt&#xff0c;选择独立交易&#xff0c; 之后使用pycharm&#xff0c;编写py文件。 from xtquant import xtdata from xtquant.xttrader import…

在QGIS中将矢量数据导出为JSON

在QGIS中将矢量数据导出为JSON的完整操作指南如下&#xff0c;支持GeoJSON标准格式及自定义配置&#xff1a; 一、标准GeoJSON导出&#xff08;推荐&#xff09; 适用场景&#xff1a;生成符合OGC标准的地理JSON文件&#xff0c;适用于Web地图开发 准备图层 确保目标图层在QG…

Netty——连接超时 与 断开重连

文章目录 1. 处理连接超时和断开重连的原因2. 处理连接超时和断开重连的方法2.1 处理连接超时2.1.1 步骤一&#xff1a;配置连接超时时间2.1.2 步骤二&#xff1a;监听连接结果 2.2 处理断开重连2.2.1 步骤一&#xff1a;监听连接断开事件2.2.2 步骤二&#xff1a;实现重连逻辑…

Redis 与 AI:从缓存到智能搜索的融合之路

Redis 与 AI&#xff1a;从缓存到智能搜索的融合之路 在当今数字化时代&#xff0c;Redis 不仅是一个高性能的缓存系统&#xff0c;更是一个强大的 AI 支持平台。Redis 通过其向量数据库功能和 AI 工具&#xff0c;为现代应用提供了独特的技术优势。 一、Redis 的 AI 能力 &…

LeetCode435 -- 预定会议问题

0. ref 参考自 1. 题目描述 预定会议问题&#xff1a;给定我们一堆区间&#xff0c;区间不能重叠&#xff08; [ 1 , 2 ] [1,2] [1,2] 和 [ 2 , 3 ] [2,3] [2,3] 的 2 2 2 不算重叠&#xff09;&#xff0c;求最多能保留多少个区间&#xff1f; 做法&#xff1a;贪心&#…

leetcode51-N皇后

leetcode 51 思路 本题可以使用回溯算法来解决。回溯算法通过尝试所有可能的解决方案来找到问题的解的算法&#xff0c;当发现当前的选择无法得到有效的解决方案时&#xff0c;就回溯到上一步&#xff0c;尝试其他的选择。对于 N 皇后问题&#xff0c;我们可以逐行放置皇后&…

linux paste 命令

paste 是 Linux 中一个用于水平合并文件内容的命令行工具&#xff0c;它将多个文件的对应行以并行方式拼接&#xff0c;默认用制表符&#xff08;Tab&#xff09;分隔。 1. 基本语法 paste [选项] 文件1 文件2 ... 2. 常用选项 选项说明-d指定拼接后的分隔符&#xff08;默…

Linux 入门:基础开发工具(上)vim,gcc/g++,make/makefile

目录 一.软件包管理器 一&#xff09;.软件包 二&#xff09;.安装软件 三&#xff09;.删除软件 二.编辑器vim 一&#xff09;.vim的基本介绍 1.正常/普通/命令模式(Normal mode) 2.插入模式(Insert mode) 3.底行模式(last line mode) 二&#xff09;.vim的基本操作 …