数据结构刷题(2)

题目:下排每个数都是先前上排那十个数在下排出现的次数

给你10分钟时间,根据上排给出十个数,在其下排填出对应的十个数
要求下排每个数都是先前上排那十个数在下排出现的次数。
上排的十个数如下:【0,1,2,3,4,5,6,7,8,9】
例子:
数值: 【 0,1,2,3,4,5,6,7,8,9 】
分配: 【 6,2,1,0,0,0,1,0,0,0 】
0在下排出现了6次,
1在下排出现了2次,
2在下排出现了1次,
3在下排出现了0次…

以此类推…
数值: 【 0,1,2,3,4,5,6,7,8,9 】
分配: 【 6,2,1,0,0,0,1,0,0,0 】
10 0 0 0 0 0 0 0 0 0
9 0 0 0 0 0 0 0 0 1
8 1 0 0 0 0 0 0 1 0
7 2 1 0 0 0 0 1 0 0
6 2 1 0 0 0 1 0 0 0
6 2 1 0 0 0 1 0 0 0

问题分析

设上排的数字用 a[i]表示, 下排的数字用b[i]表示,每排有n个数字。则应满足条件:

① ∑b[i] = n 下排所有数字的和为n

②∑a[i]*b[i] = n 上下排对应的数字乘积相加的和为n

③ 下排出现了 b[i] 个 a[i]

④b[i] 一定小于 n

方法是暴力查找,从全0开始 尝试所有的0-n的数字组合

在每次选择一个数字后,先用∑a[i]*b[i] <= n 和 ∑b[i] <= n 的条件去掉一大部分不符合条件的 相当于截枝 加快速度

然后对每次选定所有下排的数后,验证是否满足条件① 和 ③(满足这两个条件 其他的条件一定满足) 满足则返回。

实现

#include <iostream>
using namespace std;

//检查,已有数字加起来的和是否超过n
bool checkSum(int *b, int n)
{
  int sum = 0;
  for (int j = 0; j < n; j++)
  {
    sum += b[j];
  }
  return (sum == n);
}

/**
 * 检查组成是否满足:下排每个数都是先前上排那十个数在下排出现的次数。
 */
bool checkcomponet(int *a, int *b, int n)
{
  for (int i = 0; i < n; i++)
  {
    int num = 0;
    for (int j = 0; j < n; j++)
    {
      if (b[j] == a[i])
      {
        num++;
      }
    }
    if (num != b[i])
    {
      return false;
    }
  }
  return true;
}

/**
 * in:输入
 * out:输出数组
 * n:一共有多少个数字
 * num:当前要确定第几个数字
*/
bool findnum(int *in, int *out, int n, int num)
{
  bool isFind = false;
  for (int i = 0; i < n; i++)
  {
    out[num - 1] = i; //设当前数为i

    //根据和 与 积 不能大于n 去掉大部分错误的尝试
    int product = 0;
    int sum = 0;
    for (int j = num - 1; j < n; j++)
    {
      product += out[j] * in[j];
      sum += out[j];
    }
    if (sum > n || product > n)
    {
      break;
    }

    if (num == 1)
    {
      if (checkcomponet(in, out, n) && checkSum(out, n))
      {
        isFind = true;
      }
    }
    else
    {
      isFind = findnum(in, out, n, num - 1);
    }

    if (isFind == true)
    {
      break;
    }
  }
  return isFind;
}

int main()
{
  int a[11] = {0, 1, 2, 3, 4, 5, 6, 7, 8, 9};
  int b[11] = {0};
  bool isfind = findnum(a, b, 10, 10);

  cout << "a: ";
  for (int i = 0; i < 11; i++)
  {
    cout << a[i] << ' ';
  }
  cout << endl;
  if (isfind == true)
  {
    cout << "b: ";
    for (int i = 0; i < 11; i++)
    {
      cout << b[i] << ' ';
    }
    cout << endl;
  }
  else
  {
    cout << "no answer can be find!" << endl;
  }
  return 0;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

HelloHypatia

希望自己的总结对大家有所帮助!

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值