项目Euler#8系列中最大的产品

时间:2016-04-14 15:05:58

标签: c++

我正在研究Project Euler的问题#8,其中我被要求“找到这个1000位数字中具有最大产品的13个相邻数字。这个产品的价值是多少?”

这是我的C ++代码。出于某种原因,它一直给我错误的答案,我强烈怀疑它与我使用错误的数据类型有关。任何帮助将不胜感激。

{
    int n = 0;
    unsigned long long x, y;
    signed char num[] = "7316717653133062491922511967442657474235534919493496983520312774506326239578318016984801869478851843858615607891129494954595017379583319528532088055111254069874715852386305071569329096329522744304355766896648950445244523161731856403098711121722383113622298934233803081353362766142828064444866452387493035890729629049156044077239071381051585930796086670172427121883998797908792274921901699720888093776657273330010533678812202354218097512545405947522435258490771167055601360483958644670632441572215539753697817977846174064955149290862569321978468622482839722413756570560574902614079729686524145351004748216637048440319989000889524345065854122758866688116427171479924442928230863465674813919123162824586178664583591245665294765456828489128831426076900422421902267105562632111110937054421750694165896040807198403850962455444362981230987879927244284909188845801561660979191338754992005240636899125607176060588611646710940507754100225698315520005593572972571636269561882670428252483600823257530420752963450";
    while(n <= 1000)
    {
        x = (num[n] - 48) * (num[n + 1] - 48) * (num[n + 2] - 48) * (num[n + 3] - 48) * (num[n + 4] - 48) * (num[n + 5] - 48) * (num[n + 6] - 48) * (num[n + 7] - 48) * (num[n + 8] - 48) * (num[n + 9] - 48) * (num[n + 10] - 48) * (num[n + 11] - 48) * (num[n + 12] - 48);
        std::cout << "x" << x << std::endl;
        if(x > y)
        {
            y = x;
        }
        n = n + 1;
    }
    std::cout << "y" << y << std::endl;
    std::cout << "n" << n << std::endl;
}

3 个答案:

答案 0 :(得分:2)

首先,你没有像其他人在评论中所说的那样初始化y。

其次,这个表达式(num[n] - 48) * (num[n + 1] - 48) * (num[n + 2] - 48) * ...将以int精度完成,因为小于int的类型将在进行算术之前被提升为int

将其更改为

x = (unsigned long long)(num[n] - '0') * (num[n + 1] - '0') * ...

以便以unsigned long long精度计算表达式。使用&#39; 0&#39;而不是48,因为它更明确地表达了意图,并且无论'0'的值是什么,它都会起作用

当n&gt; = 1000-12时,您也有超出限制的访问权限。请改用for

int len = strlen(num);
for (n = 0; n < len - 12; n++)

如果您在每次迭代中计算产品时发现存在重叠,则可以进一步改进

答案 1 :(得分:1)

另一个答案中指出的一个主要问题是,您的结果将被视为int,超过导致错误答案的Int32的最大值。您将同时迭代1个额外索引您计算x的行可以做得更好,目前很难跟踪正在发生的事情并且很难维护。

我建议首先创建一个常量整数,它将包含您案例13中所需相邻数字的值。

const int adjacentDigits = 13;

这应该可以完成工作现在您还应该修复while循环:

while (n <= 1000 - adjacentDigits)

这里我们使用前面声明的const,所以如果我们希望代码适用于不同数量的相邻数字,我们只需更改响应的变量即可。

您应该为y

指定一个起始值
unsigned long long y = 0;

现在进入while循环,我们看到了前面提到的长线。我们可以轻松地将其编辑为:

    for (int i = n; i < n + adjacentDigits; i++)
    {
        x *= (num[i] - '0');
    }

这更容易阅读,它将删除使用铸造。现在设置变量'x'的一个问题是在外部作用域(循环外部)中声明,而不是仅仅设置变量值我们将它乘以它意味着我们需要将变量重置为0一旦我们完成了成倍增加。有两种方法可以做到这一点你可以在循环结束时将其设置为零,或者只是在我喜欢的while循环中声明它。将x的值设为1而不是0 ..

也很重要
unsigned long long x = 1;

在此之后,我们只需保留原始支票,以确定我们是否找到了新号码。

if (x > y)
{
    y = x;
}

最后我们只打印y

这是完整的代码:

const int adjacentDigits = 13;
int n = 0;
unsigned long long y = 0;
signed char num[] = "7316717653133062491922511967442657474235534919493496983520312774506326239578318016984801869478851843858615607891129494954595017379583319528532088055111254069874715852386305071569329096329522744304355766896648950445244523161731856403098711121722383113622298934233803081353362766142828064444866452387493035890729629049156044077239071381051585930796086670172427121883998797908792274921901699720888093776657273330010533678812202354218097512545405947522435258490771167055601360483958644670632441572215539753697817977846174064955149290862569321978468622482839722413756570560574902614079729686524145351004748216637048440319989000889524345065854122758866688116427171479924442928230863465674813919123162824586178664583591245665294765456828489128831426076900422421902267105562632111110937054421750694165896040807198403850962455444362981230987879927244284909188845801561660979191338754992005240636899125607176060588611646710940507754100225698315520005593572972571636269561882670428252483600823257530420752963450";
while (n <= 1000 - adjacentDigits)
{
    unsigned long long x = 1;
    for (int i = n; i < n+ adjacentDigits; i++)
    {
        x *= (num[i] - '0');
    }
    if (x > y)
    {
        y = x;
    }
    n++;
}
std::cout << y << std::endl;

您可能还想使用更合理的名称xy听起来有点像网格中使用的变量。例如,您只需更改{{1} } ymaxNumbery

答案 2 :(得分:0)

以为我会分享我的解决方案以及一个答案请求其他Scala解决方案。我不是说这在某种程度上更好,只是略有不同。另外,我认为String.substring()很慢(线性时间复杂度),所以我决定不使用它。

还包括一个处理零的小优化,其他答案没有讨论过:如果下一个(13)数字中有0,我们可以跳过所有这些数字,因为0将包含在它们的范围内同样。

import scala.annotation.tailrec

object Problem8 extends App {
  val input = """
    73167176531330624919225119674426574742355349194934
    96983520312774506326239578318016984801869478851843
    85861560789112949495459501737958331952853208805511
    12540698747158523863050715693290963295227443043557
    66896648950445244523161731856403098711121722383113
    62229893423380308135336276614282806444486645238749
    30358907296290491560440772390713810515859307960866
    70172427121883998797908792274921901699720888093776
    65727333001053367881220235421809751254540594752243
    52584907711670556013604839586446706324415722155397
    53697817977846174064955149290862569321978468622482
    83972241375657056057490261407972968652414535100474
    82166370484403199890008895243450658541227588666881
    16427171479924442928230863465674813919123162824586
    17866458359124566529476545682848912883142607690042
    24219022671055626321111109370544217506941658960408
    07198403850962455444362981230987879927244284909188
    84580156166097919133875499200524063689912560717606
    05886116467109405077541002256983155200055935729725
    71636269561882670428252483600823257530420752963450
  """

  val digits = input.filterNot(_.isWhitespace).map(_.asDigit)

  //  val adjacentDigits = 4
  val adjacentDigits = 13

  @tailrec
  def largestProduct(digits: Seq[Int],
                     largestSoFar: BigInt,
                     previousWasZero: Boolean
                    ): BigInt = {
    digits.take(adjacentDigits) match {
      case Nil        => largestSoFar
      case nextDigits =>
        val product = nextDigits.foldLeft(BigInt(1))(_ * _)
        if (product.equals(BigInt(0))) {
          val indexOfZero = if (previousWasZero) nextDigits.indexOf(0) else adjacentDigits - 1
          largestProduct(digits.drop(indexOfZero + 1), largestSoFar, true)
        } else {
          val largest = if (product > largestSoFar) product else largestSoFar
          largestProduct(digits.tail, largest, false)
        }
    }
  }

  println(largestProduct(digits, BigInt(0), true))
}