Files
freeCodeCamp/curriculum/challenges/japanese/10-coding-interview-prep/project-euler/problem-88-product-sum-numbers.md
2022-01-20 20:30:18 +01:00

3.1 KiB
Raw Permalink Blame History

id, title, challengeType, forumTopicId, dashedName
id title challengeType forumTopicId dashedName
5900f3c51000cf542c50fed6 問題 88: 積和数 5 302203 problem-88-product-sum-numbers

--description--

2 つ以上の自然数 \\{a_1, a_2, \ldots , a_k\\} からなる与えられた集合の和かつ積として表せる自然数 N は、積和数と呼ばれます。すなわち、N = a_1 + a_2 + \cdots + a_k = a_1 × a_2 × \cdots × a_k です。

例えば、6 = 1 + 2 + 3 = 1 × 2 × 3 です。

大きさが k である与えられた集合に対して、この性質を持つ最小の N を「最小積和数」と呼ぶことにします。 大きさ k = 2, 3, 4, 5, 6 の集合に対する最小積和数は次のとおりです。

k=2: 4 = 2 × 2 = 2 + 2
k=3: 6 = 1 × 2 × 3 = 1 + 2 + 3
k=4: 8 = 1 × 1 × 2 × 4 = 1 + 1 + 2 + 4
k=5: 8 = 1 × 1 × 2 × 2 × 2 = 1 + 1 + 2 + 2 + 2
k=6: 12 = 1 × 1 × 1 × 1 × 2 × 6 = 1 + 1 + 1 + 1 + 2 + 6

したがって、2 ≤ k ≤ 6 のとき、すべての最小積和数の和は 4 + 6 + 8 + 12 = 30 となりますが、8 が和の中で一度だけカウントされることに注意してください。

実際、2 ≤ k ≤ 12 の場合の最少積和数の完全集合は \\{4, 6, 8, 12, 15, 16\\} で、その和は 61 です。

2 ≤ klimit のとき、最小積和数の総和を求めなさい。

--hints--

productSumNumbers(6) は数値を返す必要があります。

assert(typeof productSumNumbers(6) === 'number');

productSumNumbers(6)30 を返す必要があります。

assert.strictEqual(productSumNumbers(6), 30);

productSumNumbers(12)61 を返す必要があります。

assert.strictEqual(productSumNumbers(12), 61);

productSumNumbers(300)12686 を返す必要があります。

assert.strictEqual(productSumNumbers(300), 12686);

productSumNumbers(6000)2125990 を返す必要があります。

assert.strictEqual(productSumNumbers(6000), 2125990);

productSumNumbers(12000)7587457 を返す必要があります。

assert.strictEqual(productSumNumbers(12000), 7587457);

--seed--

--seed-contents--

function productSumNumbers(limit) {

  return true;
}

productSumNumbers(6);

--solutions--

function productSumNumbers(limit) {
  function getProductSums(curProduct, curSum, factorsCount, start) {
    const k = curProduct - curSum + factorsCount;
    if (k <= limit) {
      if (curProduct < minimalProductSums[k]) {
        minimalProductSums[k] = curProduct;
      }
      for (let i = start; i < Math.floor((limit / curProduct) * 2) + 1; i++) {
        getProductSums(curProduct * i, curSum + i, factorsCount + 1, i);
      }
    }
  }

  const minimalProductSums = new Array(limit + 1).fill(2 * limit);
  getProductSums(1, 1, 1, 2);

  const uniqueProductSums = [...new Set(minimalProductSums.slice(2))];

  let sum = 0;
  for (let i = 0; i < uniqueProductSums.length; i++) {
    sum += uniqueProductSums[i];
  }

  return sum;
}