返回介绍

solution / 1300-1399 / 1363.Largest Multiple of Three / README

发布于 2024-06-17 01:03:20 字数 6052 浏览 0 评论 0 收藏 0

1363. 形成三的最大倍数

English Version

题目描述

给你一个整数数组 digits,你可以通过按 任意顺序 连接其中某些数字来形成 3 的倍数,请你返回所能得到的最大的 3 的倍数。

由于答案可能不在整数数据类型范围内,请以字符串形式返回答案。如果无法得到答案,请返回一个空字符串。返回的结果不应包含不必要的前导零。

 

示例 1:

输入:digits = [8,1,9]
输出:"981"

示例 2:

输入:digits = [8,6,7,1,0]
输出:"8760"

示例 3:

输入:digits = [1]
输出:""

示例 4:

输入:digits = [0,0,0,0,0,0]
输出:"0"

 

提示:

  • 1 <= digits.length <= 10^4
  • 0 <= digits[i] <= 9

解法

方法一:贪心 + 动态规划 + 逆推

我们定义 $f[i][j]$ 表示在前 $i$ 个数中选取若干个数,使得选取的数的和模 $3$ 为 $j$ 的最大长度。为了使得选取的数最大,我们需要尽可能选取更多的数,因此我们需要使得 $f[i][j]$ 尽可能大。我们初始化 $f[0][0] = 0$,其余的 $f[0][j] = -\infty$。

考虑 $f[i][j]$ 如何进行状态转移。我们可以不选取第 $i$ 个数,此时 $f[i][j] = f[i - 1][j]$;我们也可以选取第 $i$ 个数,此时 $f[i][j] = f[i - 1][(j - x_i \bmod 3 + 3) \bmod 3] + 1$,其中 $x_i$ 表示第 $i$ 个数的值。因此我们有如下的状态转移方程:

$$ f[i][j] = \max { f[i - 1][j], f[i - 1][(j - x_i \bmod 3 + 3) \bmod 3] + 1 } $$

如果 $f[n][0] \le 0$,那么我们无法选取任何数,因此答案字符串为空。否则我们可以通过 $f$ 数组进行逆推,找出选取的数。

定义 $i = n$, $j = 0$,从 $f[i][j]$ 开始逆推,记 $k = (j - x_i \bmod 3 + 3) \bmod 3$,如果 $f[i - 1][k] + 1 = f[i][j]$,那么我们选取了第 $i$ 个数,否则我们没有选取第 $i$ 个数。如果我们选取了第 $i$ 个数,那么我们将 $j$ 更新为 $k$,否则我们保持 $j$ 不变。为了使得同等长度的数最大,我们应该优先选取较大的数,因此,我们在前面首先对数组进行排序。

时间复杂度 $O(n \times \log n)$,空间复杂度 $O(n)$。其中 $n$ 为数组的长度。

class Solution:
  def largestMultipleOfThree(self, digits: List[int]) -> str:
    digits.sort()
    n = len(digits)
    f = [[-inf] * 3 for _ in range(n + 1)]
    f[0][0] = 0
    for i, x in enumerate(digits, 1):
      for j in range(3):
        f[i][j] = max(f[i - 1][j], f[i - 1][(j - x % 3 + 3) % 3] + 1)
    if f[n][0] <= 0:
      return ""
    arr = []
    j = 0
    for i in range(n, 0, -1):
      k = (j - digits[i - 1] % 3 + 3) % 3
      if f[i - 1][k] + 1 == f[i][j]:
        arr.append(digits[i - 1])
        j = k
    i = 0
    while i < len(arr) - 1 and arr[i] == 0:
      i += 1
    return "".join(map(str, arr[i:]))
class Solution {
  public String largestMultipleOfThree(int[] digits) {
    Arrays.sort(digits);
    int n = digits.length;
    int[][] f = new int[n + 1][3];
    final int inf = 1 << 30;
    for (var g : f) {
      Arrays.fill(g, -inf);
    }
    f[0][0] = 0;
    for (int i = 1; i <= n; ++i) {
      for (int j = 0; j < 3; ++j) {
        f[i][j] = Math.max(f[i - 1][j], f[i - 1][(j - digits[i - 1] % 3 + 3) % 3] + 1);
      }
    }
    if (f[n][0] <= 0) {
      return "";
    }
    StringBuilder sb = new StringBuilder();
    for (int i = n, j = 0; i > 0; --i) {
      int k = (j - digits[i - 1] % 3 + 3) % 3;
      if (f[i - 1][k] + 1 == f[i][j]) {
        sb.append(digits[i - 1]);
        j = k;
      }
    }
    int i = 0;
    while (i < sb.length() - 1 && sb.charAt(i) == '0') {
      ++i;
    }
    return sb.substring(i);
  }
}
class Solution {
public:
  string largestMultipleOfThree(vector<int>& digits) {
    sort(digits.begin(), digits.end());
    int n = digits.size();
    int f[n + 1][3];
    memset(f, -0x3f, sizeof(f));
    f[0][0] = 0;
    for (int i = 1; i <= n; ++i) {
      for (int j = 0; j < 3; ++j) {
        f[i][j] = max(f[i - 1][j], f[i - 1][(j - digits[i - 1] % 3 + 3) % 3] + 1);
      }
    }
    if (f[n][0] <= 0) {
      return "";
    }
    string ans;
    for (int i = n, j = 0; i; --i) {
      int k = (j - digits[i - 1] % 3 + 3) % 3;
      if (f[i - 1][k] + 1 == f[i][j]) {
        ans += digits[i - 1] + '0';
        j = k;
      }
    }
    int i = 0;
    while (i < ans.size() - 1 && ans[i] == '0') {
      ++i;
    }
    return ans.substr(i);
  }
};
func largestMultipleOfThree(digits []int) string {
  sort.Ints(digits)
  n := len(digits)
  const inf = 1 << 30
  f := make([][]int, n+1)
  for i := range f {
    f[i] = make([]int, 3)
    for j := range f[i] {
      f[i][j] = -inf
    }
  }
  f[0][0] = 0
  for i := 1; i <= n; i++ {
    for j := 0; j < 3; j++ {
      f[i][j] = max(f[i-1][j], f[i-1][(j-digits[i-1]%3+3)%3]+1)
    }
  }
  if f[n][0] <= 0 {
    return ""
  }
  ans := []byte{}
  for i, j := n, 0; i > 0; i-- {
    k := (j - digits[i-1]%3 + 3) % 3
    if f[i][j] == f[i-1][k]+1 {
      ans = append(ans, byte('0'+digits[i-1]))
      j = k
    }
  }
  i := 0
  for i < len(ans)-1 && ans[i] == '0' {
    i++
  }
  return string(ans[i:])
}
function largestMultipleOfThree(digits: number[]): string {
  digits.sort((a, b) => a - b);
  const n = digits.length;
  const f: number[][] = new Array(n + 1).fill(0).map(() => new Array(3).fill(-Infinity));
  f[0][0] = 0;
  for (let i = 1; i <= n; ++i) {
    for (let j = 0; j < 3; ++j) {
      f[i][j] = Math.max(f[i - 1][j], f[i - 1][(j - (digits[i - 1] % 3) + 3) % 3] + 1);
    }
  }
  if (f[n][0] <= 0) {
    return '';
  }
  const arr: number[] = [];
  for (let i = n, j = 0; i; --i) {
    const k = (j - (digits[i - 1] % 3) + 3) % 3;
    if (f[i - 1][k] + 1 === f[i][j]) {
      arr.push(digits[i - 1]);
      j = k;
    }
  }
  let i = 0;
  while (i < arr.length - 1 && arr[i] === 0) {
    ++i;
  }
  return arr.slice(i).join('');
}

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。
列表为空,暂无数据
    我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
    原文