每天一道C语言编程:排队买票

题目描述

有M个小孩到公园玩,门票是1元。其中N个小孩带的钱为1元,K个小孩带的钱为2元。售票员没有零钱,问这些小孩共有多少种排队方法,使得售票员总能找得开零钱。注意:两个拿一元零钱的小孩,他们的位置互换,也算是一种新的排法。(M<=10)

输入格式

输入一行,M,N,K(其中M=N+K,M<=10).

输出格式

输出一行,总的排队方案。

样例输入

4 2 2

样例输出

8
方法一:

题目分析:
由题目可知,必须满足条件N>=K,拆分这个条件:

N=K

N个小孩带的钱为1元,另外N个小孩带的钱为2元,即2N=M,可以直接用卡特兰数:

由于题目中说小孩交换位置算一种新的排队方式,所以还要再乘上 n 的全排列(乘两遍:N个小孩带的钱为1元,另外N个小孩带的钱为2元),即

K(n)=\frac{C_{2n}^{n}}{n+1}\times n!\times n!

N>K 

先将非法的排列方法筛选出来,再用总的排列方法-非法的排列方法,得最终排列的方法数:

总的排列方法:
因为由M人,所以总的排列方法有M!

非法的排列可以将其分为三个部分:

  • 前 2P 个小孩
  • 第 2P + 1 个小孩
  • 剩下的小孩,假设共 R 个(R = M - 2P - 1)

第一部分可以使用卡特兰公式进行运算,即,其中p在0~k范围内,但不能为K,因为第二部分必须为K中的一部分

其次,p可以为0,因为第二部分2p+1为持有2元的小孩,即第一个排队的人就是持有2元的小孩,也是非法的排列顺序:

\sum_{P=0}^{K-1}K(P)A_{N}^{P}A_{K}^{P}

由于第一部分已经用了k中的p个,第二部分有k-p个选择:

K-P

 第三部分随意排列,即r=m-2*p-1进行随意排列:

R!

所以合法的排列公式为:

M!-\sum_{P=0}^{K-1}K(P)A_{N}^{P}A_{K}^{P}(K-P)R!

用代码实现即:

 long long sum = 0;for (int p = 0; p <k; p++) {int r = m - 2 * p - 1;long long fail = catalan(p) * rank(k, p) * rank(n, p) * (k - p) * rank(r, r);sum += fail;}long long result = rank(m, m) - sum;printf("%lld\n", result);

所以完整的代码得,如果其中有一些小漏洞,请大佬们不吝赐教!💖💖

#include<stdio.h>// 求排列数
long long rank(int a1, int a2) {if (a2 == 0) return 1;a2--;//这里一定不能忘记,因为要排除rank(c1,c2)中,c1为0得情况long long pro = a1;for (int i = 0; i < a2; i++) {a1--;pro *= a1;}return pro;
}// 求组合数
long long comb(int c1, int c2) {return rank(c1, c2) / rank(c2, c2);
}// 求卡特兰数
long long catalan(int n) {return comb(2 * n, n) / (n + 1);
}int main() {int m, n, k;scanf("%d %d %d", &m, &n, &k);if (n < k) {printf("Error: n must be greater than or equal to k.\n");return 0;}else {long long sum = 0;for (int p = 0; p <k; p++) {int r = m - 2 * p - 1;long long fail = catalan(p) * rank(k, p) * rank(n, p) * (k - p) * rank(r, r);sum += fail;}long long result = rank(m, m) - sum;printf("%lld\n", result);}return 0;
}
方法二

我在网上也看到一种更简便得方法,在这里分享给大家,我的理解如下:

他利用了数据结构中的next_permutation(a,a+N)方法进行全排列,不熟悉的可以看这篇文章:

http://t.csdn.cn/TREc9

我们可以把它理解为序列的字典序的前后,严格来讲,就是对于当前序列pn,他的下一个序列pn+1满足:不存在另外的序列pm,使pn<pm<pn+1

正确的排列方法为:每个2前面至少对应着一个1

用一个num记录,num如果有1就++,有2就- - ,如果过程中num<0则证明存在有一个2没有一个1对应。

代码为:

这里初始前面的1为0~K-1,后面的2为K~N-1 所以小于K的就相当于是1了,大于K的就相当于是2了,为了方便使用next_permutation()

 for (int i=0; i<K; i++) {a[i] = i;}for (int i=K; i<N; i++) {a[i] = i;}do {int flag = 0;// check每个全排列, num务必要初始化 int num = 0;for (int i=0; i<N; i++) {if (a[i] >= K) {num--;} else {num++;}if (num < 0) {flag = 1;break;}}if (flag == 0) {
//                for (int i=0; i<N; i++) {
//                    cout << a[i];
//                }
//                cout << "\n";res++;}

这还不够,需要判断类似1122的所有全排列包括重复的,因为初始前面的1为0~K-1,后面的2为K~N-1 ,对于0123的序列我们可以直接使用next_permutation(),完整代码如下:

#include <iostream>
#include <algorithm>
using namespace std;
int N, K, M;
int main() {while (cin >> N >> K >> M) {int a[N], res = 0;for (int i=0; i<K; i++) {a[i] = i;}for (int i=K; i<N; i++) {a[i] = i;}do {int flag = 0;// check每个全排列, num务必要初始化 int num = 0;for (int i=0; i<N; i++) {if (a[i] >= K) {num--;} else {num++;}if (num < 0) {flag = 1;break;}}if (flag == 0) {
//                for (int i=0; i<N; i++) {
//                    cout << a[i];
//                }
//                cout << "\n";res++;}} while (next_permutation(a,a+N));cout << res << "\n";}return 0;
}

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

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

相关文章

GoFrame v2.5 版本发布,企业级 Golang 开发框架

大家好啊&#xff0c;GoFrame 框架今天发布了 v2.5.0 正式版本啦&#xff01;&#x1f44f;&#x1f44f;&#x1f44f;&#x1f44f; 本次版本主要是对已有功能组件以及开发工具上的改进工作。其中&#xff0c;开发工具新增了 gf gen ctrl 命令&#xff0c;以规范化定义、开发…

orcle报错:TNS 监听程序无法为请求的服务器类型找到可用的处理程序

orcle报错&#xff1a;TNS 监听程序无法为请求的服务器类型找到可用的处理程序 方法一&#xff1a;配置文件修改 服务端的数据库是专用服务器,但是在客户端的tnsname.ora里配置中设置了连接方式为shared,这种情况下打开tnsnames.ora, 找到安装orcle的安装目录&#xff0c;点…

Java常用类(一)

⭐ 基本数据类型的包装类⭐ 包装类基本知识⭐ 包装类的用途⭐ 自动装箱和拆箱⭐ 自定义一个简单的包装类 ⭐ 字符串相关类 ⭐ 基本数据类型的包装类 我们之前写过八种基本数据类型并不是对象&#xff0c;为了将基本类型数据和对象之间实现互相转化&#xff0c;Java 为每一个基…

【EXCEL】数据录入的快捷键和正确格式

目录 0.环境 1.内容概述 2.具体内容 2.1数据录入换行换列的快捷键&#xff08;标准的数据输入方式&#xff09; 2.2日期的正确格式和使用&#xff08;标准日期格式与长日期&#xff09; 2.2.1 标准日期 2.2.2 长日期 2.2.3 显示当前日期和时间的快捷键 2.3百分比的正确…

stm32(串口知识点)

HAL串口发送/接收函数&#xff1a; HAL_UART_Transmit(); 串口发送数据&#xff0c;使用超时管理机制HAL_UART_Receive(); 串口接收数据&#xff0c;使用超时管理机制HAL_UART_Transmit_IT(); 串口中断模式发送 HAL_UART_Receive_IT(); 串口中断模式接收 HAL_UART_Transmit(…

人工智能LLM模型:奖励模型的训练、PPO 强化学习的训练、RLHF

人工智能LLM模型&#xff1a;奖励模型的训练、PPO 强化学习的训练 1.奖励模型的训练 1.1大语言模型中奖励模型的概念 在大语言模型完成 SFT 监督微调后&#xff0c;下一阶段是构建一个奖励模型来对问答对作出得分评价。奖励模型源于强化学习中的奖励函数&#xff0c;能对当前…

数据库三范式

MySQL系列文章 MySQL&#xff08;一&#xff09;基本架构、SQL语句操作、试图 MySQL&#xff08;二&#xff09;索引原理以及优化 MySQL&#xff08;三&#xff09;SQL优化、Buffer pool、Change buffer MySQL&#xff08;四&#xff09;事务原理及分析 MySQL&#xff08;五&a…

LoggerFactory is not a Logback LoggerContext but Logback is on the classpath

springboot项目报错如下&#xff1a; 这个错误是由于在你的Java代码中使用了Logback日志库&#xff0c;但是同时又存在与Logback竞争的其他日志库&#xff08;例如slf4j-simple&#xff09;导致的冲突。 要解决这个问题&#xff0c;你可以尝试以下几个步骤&#xff1a; 1. 检…

设计模式——享元模式

享元模式 定义 享元模式&#xff08;Flyweight Pattern&#xff09;是池技术的重要实现方式。 使用共享对象可以有效地支持大量的细粒度对象。 优缺点、应用场景 优点 可以大大减少应用程序创建对象的数量&#xff0c;降低程序内存占用。 缺点 提高了系统的复杂度&…

【杨氏矩阵】

这篇文章的对应思维导图为&#xff1a;思维导图 思维导图对应代码&#xff1a; //杨氏矩阵 #include<stdio.h>//void ysjz1(int a[3][3],int k) { // int x 0; // int y 2; // while (x < 2 && y > 0) { // if (a[x][y] > k) { // y--; // } // …

微服务系列文章 之SpringBoot之定时任务详解

序言 使用SpringBoot创建定时任务非常简单&#xff0c;目前主要有以下三种创建方式&#xff1a; 一、基于注解(Scheduled)二、基于接口&#xff08;SchedulingConfigurer&#xff09; 前者相信大家都很熟悉&#xff0c;但是实际使用中我们往往想从数据库中读取指定时间来动态…

机器学习-线性代数-5-空间中的向量投影与最小二乘法

空间中的向量投影与最小二乘法 文章目录 空间中的向量投影与最小二乘法一、引入二、投影和投影的描述1、投影描述最近2、利用矩阵描述投影(1)向一维直线投影(2)向二维平面投影(3)向n维子空间投影的一般情况 三、最小二乘法1、重要的子空间(1)互补的子空间(2)正交的子空间(3)相互…