Permutation

2017-12-11  本文已影响0人  好之者不如乐之者

这是一道要求你用1~9这十个数字来构造三个三位数abc,def,ghi。使得abc是def的二分之一,同时是ghi的三分之一。打印出所有满足条件的三个三位数。
我先想到的是以下的法一:

#include <cstdio>

int main()
{
  for (int a = 1; a < 10; a++) {
    for (int b = 1; b < 10; b++) {
      for (int c = 1; c < 10; c++) {
        for (int d = 1; d < 10; d++) {
         for (int e = 1; e < 10; e++) {
           for (int f = 1; f < 10; f++) {
            for (int g = 1; g < 10; g++) {
              for (int h = 1; h < 10; h++) {
                if (b != a && c != a && c != b && d != a && d != b && d != c && e != a && e != b && e != c && e != d && f != a && f != b && f != c && f != d && f != e && g != a && g != b && g != c && g != d && g != e && g != f && h != a && h != b && h != c && h != d && h != e && h != f && h != g) {
                  int i = 45-a-b-c-d-e-f-g-h;
                  int sum1 = 100*a + 10 * b + c, sum2 = 100 * d + 10 * e + f, sum3 = 100 * g + 10 * h + i;
                  if (2 * sum1 == sum2 && 3 * sum1 == sum3)
                   printf("%d %d %d\n", sum1, sum2, sum3);
                }
              }
            }
          }
        }
      }
    }
  }
  return 0;
}

但是这个程序有一个明显的问题就是速度太慢,整个程序运行了0.368ms因为这是在所有循环之后再判断是否成立(即有没有数重复使用),这样就会浪费很多次循环,其次三个三位数的首位:a、d、g是可以有更明确的范围限定的,因为假如a超过了3,那么ghi一定大于1000了。还有就是我一开始把循环写成了形如for (int i = 1; i < 10; && i != j; i++) 而这种写法其实是错误的,因为本来这样写都主旨在于对i != j 所有其它情况进行循环,但假如这样写的话,i只会循环到最大的i使得i<j。这样就会有很多情况讨论不到。
所以我之后就写了以下的代码:

#include <cstdio>

int main()
{
  for (int a = 1; a < 4; a++) {
    for (int b = 1; b < 10; b++) {
      if (b == a)
        continue;
      for (int c = 1; c < 10; c++) {
        if (c == a || c == b)
          continue;
        for (int d = 2; d < 7; d++) {
          if (d == a || d == b || d == c)
            continue;
          for (int e = 1; e < 10; e++) {
            if (e == a || e == b || e == c || e == d)
              continue;
            for (int f = 1; f < 10; f++) {
              if (f == a || f == b || f == c || f == d || f == e)
                continue;
              for (int g = 3; g < 10; g++) {
                if (g == a || g == b || g == c || g == d || g == e || g == f)
                  continue;
                for (int h = 1; h < 10; h++) {
                  if (h == a || h == b || h == c || h == d || h == e || h == f || h == g)
                    continue;
                  int i = 45-a-b-c-d-e-f-g-h;
                  int x = 100 * a + 10 * b + c, y = 100 * d + 10 * e + f, z = 100 * g + 10 * h + i;
                  if (2 * x == y && 3 * x == z)
                    printf("%d %d %d\n", x, y, z);
                }
              }
          }
        }
      }
    }
  }
  return 0;
}

这样子,速度一下字就增加到0.08ms了,这比之前的速度快了整整46倍!但是,我经过反思,又觉得循环还能减少到三个,以下就是这种方法的代码:

#include <cstdio>

int main()
{
  for (int a = 1; a < 4; a++) {
    for (int b = 1; b < 10; b++) {
      if (b == a)
        continue;
      for (int c = 1; c < 10; c++) {
        if (c == a || c == b)
          continue;
        int x = 100 * a + 10 * b + c;
        int y = 2 * x, z = 3 * x;
        if (z > 987)
          break;
        int digits[10] = {0}, ok = 1;
        digits[0] = digits[a] = digits[b] = digits[c] = 1;
        while (y != 0 && z != 0) {
          int i = y % 10, j = z % 10;
          if (i == j || digits[i] || digits[j]) {
            ok = 0;
            break;
          }
          y /= 10;
          z /= 10;
          digits[i] = digits[j] = 1;
        }
        if (ok)
          printf("%d %d %d\n", x, 2 * x, 3 * x);
      }
    }
  }
  return 0;
}

而这种方法的速度已经增快到0.01ms了。有些人可能会不明白一维数组digits的作用是什么,我就在此解释一下吧。digits赋值为以的位置表示这个位置上的数字已被占用,比如digits[1] == 1表示数字一已经被使用了。而while循环里的if判断的意义是看i和j是不是相等,并看digits中的第i个与第j个是否为0,以判断有没有重复数字使用。
以下是法四,供参考:

#include <cstdio>

void get_next_digit(int digits[], int numbers[])
{
    int total = 0;
    for (int i = 1; i < 10; i++)
        total += digits[i];

    // full
    if (total == 8) {
        total = 0;
        for (int i = 0; i < 8; i++)
            total += numbers[i];
        int x = numbers[0]*100 + numbers[1]*10 + numbers[2];
        int y = numbers[3]*100 + numbers[4]*10 + numbers[5];
        int z = numbers[6]*100 + numbers[7]*10 + 45 - total;
        if (2*x == y && 3*x == z)
            printf("%d %d %d\n", x, y, z);
        return;
    }

    // not full yet
    for (int i = 1; i < 10; i++) {
        if (digits[i])
            continue;
        digits[i] = 1;
        numbers[total] = i;
        get_next_digit(digits, numbers);
        digits[i] = 0;
        numbers[total] = 0;
    }
}

int main()
{
#ifndef CON_IO
    freopen("permutation.in", "r", stdin);
    freopen("permutation.out", "w", stdout);
#endif
    int digits[10] = {0}, numbers[9] = {0};
    digits[0] = 1;
    get_next_digit(digits, numbers);
    return 0;
}
上一篇 下一篇

猜你喜欢

热点阅读