• 每天一道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!

    用代码实现即:

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

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

    1. #include<stdio.h>
    2. // 求排列数
    3. long long rank(int a1, int a2) {
    4. if (a2 == 0)
    5. return 1;
    6. a2--;//这里一定不能忘记,因为要排除rank(c1,c2)中,c10得情况
    7. long long pro = a1;
    8. for (int i = 0; i < a2; i++) {
    9. a1--;
    10. pro *= a1;
    11. }
    12. return pro;
    13. }
    14. // 求组合数
    15. long long comb(int c1, int c2) {
    16. return rank(c1, c2) / rank(c2, c2);
    17. }
    18. // 求卡特兰数
    19. long long catalan(int n) {
    20. return comb(2 * n, n) / (n + 1);
    21. }
    22. int main() {
    23. int m, n, k;
    24. scanf("%d %d %d", &m, &n, &k);
    25. if (n < k) {
    26. printf("Error: n must be greater than or equal to k.\n");
    27. return 0;
    28. }
    29. else {
    30. long long sum = 0;
    31. for (int p = 0; p <k; p++) {
    32. int r = m - 2 * p - 1;
    33. long long fail = catalan(p) * rank(k, p) * rank(n, p) * (k - p) * rank(r, r);
    34. sum += fail;
    35. }
    36. long long result = rank(m, m) - sum;
    37. printf("%lld\n", result);
    38. }
    39. return 0;
    40. }
    方法二

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

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

    http://t.csdn.cn/TREc9

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

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

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

    代码为:

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

    1. for (int i=0; i<K; i++) {
    2. a[i] = i;
    3. }
    4. for (int i=K; i<N; i++) {
    5. a[i] = i;
    6. }
    7. do {
    8. int flag = 0;
    9. // check每个全排列, num务必要初始化
    10. int num = 0;
    11. for (int i=0; i<N; i++) {
    12. if (a[i] >= K) {
    13. num--;
    14. } else {
    15. num++;
    16. }
    17. if (num < 0) {
    18. flag = 1;
    19. break;
    20. }
    21. }
    22. if (flag == 0) {
    23. // for (int i=0; i<N; i++) {
    24. // cout << a[i];
    25. // }
    26. // cout << "\n";
    27. res++;
    28. }

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

    1. #include <iostream>
    2. #include <algorithm>
    3. using namespace std;
    4. int N, K, M;
    5. int main() {
    6. while (cin >> N >> K >> M) {
    7. int a[N], res = 0;
    8. for (int i=0; i<K; i++) {
    9. a[i] = i;
    10. }
    11. for (int i=K; i<N; i++) {
    12. a[i] = i;
    13. }
    14. do {
    15. int flag = 0;
    16. // check每个全排列, num务必要初始化
    17. int num = 0;
    18. for (int i=0; i<N; i++) {
    19. if (a[i] >= K) {
    20. num--;
    21. } else {
    22. num++;
    23. }
    24. if (num < 0) {
    25. flag = 1;
    26. break;
    27. }
    28. }
    29. if (flag == 0) {
    30. // for (int i=0; i<N; i++) {
    31. // cout << a[i];
    32. // }
    33. // cout << "\n";
    34. res++;
    35. }
    36. } while (next_permutation(a,a+N));
    37. cout << res << "\n";
    38. }
    39. return 0;
    40. }

  • 相关阅读:
    Java 之JSch实现ssh远程操作
    Python数据攻略-离群值的5种常用处理方法和可视化
    Mysql---第四篇
    【Unity】进度条和血条的三种做法
    43道Python经典案例题(有答案)
    JAVA个人博客系统设计与实现 毕业设计开题报告
    PCIe系列专题之二:2.6 Flow Control初始化
    java排序之compareTo方法和Comparator<T>接口
    高性能系统架构设计之:多级缓存
    从0到1学会Git(第三部分):Git的远程仓库链接与操作
  • 原文地址:https://blog.csdn.net/weixin_69884785/article/details/131770286