7

我正在尝试解决以下问题:

找到最小的 n 位整数 c,它有 k 个 1 位,并且是两个 n 位整数的和,其中 g、h 位设置为 1。 g, h, k <= n

首先,这里的n位整数意味着我们可以使用所有n位,即最大值。这样一个整数的值为2^n - 1。所描述的整数可能根本不存在。很明显,这种情况k > g + h没有解决方案,g + h = k答案就是2^k - 1(第一位k是 1 位,k - n前面是零)。

至于程序应该做什么的一些例子:

g = h = k = 4, n = 10 :
0000001111 + 0000001111 = 0000011110
15 + 15 = 30 (30 should be the output)


(4, 6, 5, 10):
0000011110 + 0000111111 = 0001011101
30 + 63 = 93

(30, 1, 1, 31):
1 + (2^30 - 1) = 2^30

在我看来,这是一个动态规划问题,我选择了以下方法:让dp[g][h][k][n][c]是所描述的整数,并且c是用于携带的可选位。我尝试根据最低位重建可能的总和。所以,dp[g][h][k][n + 1][0]是最小的

(0, 0):       dp[g][h][k][n][0]
(0, 0): 2^n + dp[g][h][k - 1][n][1]
(1, 0): 2^n + dp[g - 1][h][k - 1][n][0]
(0, 1): 2^n + dp[g][h - 1][k - 1][n][0]

同样,dp[g][h][k][n + 1][1]是最小值

(1, 1): dp[g - 1][h - 1][k][n][0]
(1, 1): dp[g - 1][h - 1][k - 1][n][1] + 2^n
(1, 0): dp[g - 1][h][k][n][1]
(0, 1): dp[g][h - 1][k][n][1]

这个想法并不难,但我对这些事情并没有真正的经验,而且我的算法即使对于最简单的情况也不起作用。我选择了自上而下的方法。我很难考虑所有极端情况。我真的不知道我是否正确选择了递归基础等。我的算法甚至不适用于最基本的情况g = h = k = 1, n = 2(答案是01 + 01 = 10)。不应该有答案,g = h = k = 1, n = 1但是算法给出了1(这基本上是前一个示例输出1而不是 的原因2)。所以,这是我糟糕的代码(只有非常基本的 C++):

int solve(int g, int h, int k, int n, int c = 0) {
  if (n <= 0) {
    return 0;
  }
  if (dp[g][h][k][n][c]) {
    return dp[g][h][k][n][c];
  }
  if (!c) {
    if (g + h == k) {
      return dp[g][h][k][n][c] = (1 << k) - 1;
    }
    int min, a1, a2, a3, a4;
    min = a1 = a2 = a3 = a4 = std::numeric_limits<int>::max();
    if (g + h > k && k <= n - 1) {
      a1 = solve(g, h, k, n - 1, 0);
    }
    if (g + h >= k - 1 && k - 1 <= n - 1) {
      a2 = (1 << (n - 1)) + solve(g, h, k - 1, n - 1, 1);
    }
    if (g - 1 + h >= k - 1 && k - 1 <= n - 1) {
      a3 = (1 << (n - 1)) + solve(g - 1, h, k - 1, n - 1, 0);
    }
    if (g + h - 1 >= k - 1 && k - 1 <= n - 1) {
      a4 = (1 << (n - 1)) + solve(g, h - 1, k - 1, n - 1, 0);
    }
    min = std::min({a1, a2, a3, a4});
    return dp[g][h][k][n][c] = min;
  } else {
    int min, a1, a2, a3, a4;
    min = a1 = a2 = a3 = a4 = std::numeric_limits<int>::max();
    if (g - 2 + h >= k && k <= n - 1) {
      a1 = solve(g - 1, h - 1, k, n - 1, 0);
    }
    if (g - 2 + h >= k - 1 && k - 1 <= n - 1) {
      a2 = (1 << (n - 1)) + solve(g - 1, h - 1, k - 1, n - 1, 1);
    }
    if (g - 1 + h >= k && k <= n - 1) {
      a3 = solve(g - 1, h, k, n - 1, 1);
    }
    if (g - 1 + h >= k && k <= n - 1) {
      a4 = solve(g, h - 1, k, n - 1, 1);
    }
    min = std::min({a1, a2, a3, a4});
    return dp[g][h][k][n][c] = min;
  }
}
4

2 回答 2

4

您可以根据位数 g、h 和 k 构造最小的和,而无需进行任何动态编程。假设 g ≥ h(否则切换它们)这些是规则:

k≤h≤g

      11111111  <-  g ones  
  111100000111  <-  h-k ones + g-k zeros + k ones
 1000000000110  <-  n must be at least h+g-k+1

h≤k≤g

    1111111111  <-  g ones  
      11111100  <-  h ones + k-h zeros
    1011111011  <-  n must be at least g+1

h≤g≤k

 1111111100000  <-  g ones + k-g ones  
 1100000011111  <-  g+h-k ones, k-h zeros, k-g ones
11011111111111  <-  n must be at least k+1, or k if g+h=k

示例:n=10、g=6 和 h=4 时的所有 k 值:

k=1           k=2           k=3           k=4       
0000111111    0000111111    0000111111    0000111111
0111000001    0011000011    0001000111    0000001111
----------    ----------    ----------    ----------
1000000000    0100000010    0010000110    0001001110
k=4           k=5           k=6
0000111111    0000111111    0000111111
0000001111    0000011110    0000111100
----------    ----------    ----------
0001001110    0001011101    0001111011
k=6           k=7           k=8           k=9           k=10
0000111111    0001111110    0011111100    0111111000    1111110000
0000111100    0001110001    0011000011    0100000111    0000001111
----------    ----------    ----------    ----------    ----------
0001111011    0011101111    0110111111    1011111111    1111111111

或者,不先计算 a 和 b 就直接计算 c 的值:

k≤h≤g

c = (1 << (g + h - k)) + ((1 << k) - 2)

h≤k≤g

c = (1 << g) + ((1 << k) - 1) - (1 << (k - h))

h≤g≤k

c = ((1 << (k + 1)) - 1) - (1 << ((g - h) + 2 * (k - g)))

h + g = k

c = (1 << k) - 1

这导致了这个令人失望的平凡代码:

int smallest_sum(unsigned n, unsigned g, unsigned h, unsigned k) {
    if (g < h) {unsigned swap = g; g = h; h = swap;}
    if (k == 0) return (g > 0 || h > 0 || n < 1) ? -1 : 0;
    if (h == 0) return (g != k || n < k) ? -1 : (1 << k) - 1;
    if (k <= h) return (n <= h + g - k) ? -1 : (1 << (g + h - k)) + ((1 << k) - 2);
    if (k <= g) return (n <= g) ? -1 : (1 << g) + ((1 << k) - 1) - (1 << (k - h));
    if (k < g + h) return (n <= k) ? -1 : (1 << (k + 1)) - 1 - (1 << (2 * k - g - h));
    if (k == g + h) return (n < k) ? -1 : (1 << k) - 1;
    return -1;
}

一些示例结果:

n=31, g=15, h=25, k=10  ->  1,073,742,846  (1000000000000000000001111111110)
n=31, g=15, h=25, k=20  ->     34,602,975  (0000010000011111111111111011111)
n=31, g=15, h=25, k=30  ->  2,146,435,071  (1111111111011111111111111111111)

(我将 n、g、h 和 k 从 0 到 20 的每个值与蛮力算法的结果进行了比较,以检查正确性,发现没有差异。)

于 2017-09-30T04:51:13.457 回答
2

我不太相信动态编程方法。如果我理解正确,您将需要根据先前的计算定义如何在有和没有进位位的情况下转到dp[g + 1][h][k][n]、和dp[g][h + 1][k][n],我不确定所有这些的正确规则是什么。dp[g][h][k + 1][n]dp[g][h][k][n + 1]

我认为一个更简单的方式来考虑这个问题是一个A* 搜索树,其中每个节点包含两个要添加的部分候选数字,我们称它们为 G 和 H。你从一个 G = 0 和 H = 0 的节点开始级别 m = 0,工作如下:

  1. 如果 G + H 有 n 或更少的位和 k 1 位,那就是解决方案,你找到了!
  2. 否则,如果
    n - m < G + H - k 中 1 的位数,则
    丢弃该节点(无法解决)。
  3. 否则,如果
    (g + h) - (G 中的 1 位数 + H 中的 1 位数) < k - G + H 中的 1 位数,
    则丢弃节点(不可行的候选者)。
  4. 否则,将节点分支到新级别。通常,每个节点最多可构成四个子节点,分别在 G 和 H 前面加上 0 和 0、0 和 1、1 和 0 或 1 和 1。然而:
    • 如果 G 中 1 的位数少于 g,则只能在 G 前面加上 1,对于 H 和 h 也是如此。

    • 在级别 m(G 和 H 有 m 位),如果n - m > g - G 中 1 的位数,您只能在 G 前面加上 0
      ,对于 H 和 h 也是如此。
    • 如果 G == H 和 g == h,您可以跳过 0 和 1 以及 1 和 0 之一,因为它们将导致相同的子树。
  5. 继续下一个节点并重复,直到找到解决方案或者您没有更多节点可以访问。

您访问节点的顺序很重要。您应该将节点存储在优先级队列/堆中,以便下一个节点始终是可能导致最佳解决方案的第一个节点。这实际上很简单,您只需要为每个节点 G + H 取前缀,并在其前面加上必要数量的 1 位即可达到 k;这是从那里最好的解决方案。

丢弃无效节点可能有更好的规则(步骤 2 和 3),但算法的思想是相同的。

于 2017-09-26T13:59:51.550 回答