当前位置: 首页 > 面试题库 >

使用Scala查找质数。帮我改善

辛星宇
2023-03-14
问题内容

我写了这段代码来查找小于scala中给定数字i的质数。

def findPrime(i : Int) : List[Int] = i match {
    case 2 => List(2)
    case _ => {
    val primeList = findPrime(i-1)
    if(isPrime(i, primeList)) i :: primeList else primeList
    }
}

def isPrime(num : Int, prePrimes : List[Int]) : Boolean = prePrimes.forall(num % _ != 0)

但是,我感觉到了findPrime函数,尤其是这一部分:

case _ => {
    val primeList = findPrime(i-1)
    if(isPrime(i, primeList)) i :: primeList else primeList
}

不太符合功能风格。

我仍在学习函数式编程。谁能帮助我改善此代码,使其更具功能。

非常感谢。


问题答案:

样式在我看来不错。尽管Eratosthenes筛网是查找质数的一种非常有效的方法,但是您的方法也很有效,因为您仅测试与已知质数的除法。但是,您需要当心-
您的递归函数不是尾递归。尾部递归函数不会修改递归调用的结果-
在您的示例中,您将优先执行递归调用的结果。这意味着您将拥有一个长调用堆栈,因此findPrime将不适用于大型i。这是尾递归的解决方案。

def primesUnder(n: Int): List[Int] = {
  require(n >= 2)

  def rec(i: Int, primes: List[Int]): List[Int] = {
    if (i >= n) primes
    else if (prime(i, primes)) rec(i + 1, i :: primes)
    else rec(i + 1, primes)
  }

  rec(2, List()).reverse
}

def prime(num: Int, factors: List[Int]): Boolean = factors.forall(num % _ != 0)

这个解决方案并不漂亮-
它是使您的解决方案适用于大参数的更多细节。由于列表是向后构建的,因此可以利用快速添加的内容,因此需要将列表反转。作为替代方案,你可以使用一个ArrayVectorListBuffer对结果追加。Array但是,使用时,您需要估计要为其分配多少内存。幸运的是,我们知道它pi(n)大约等于,n / ln(n)因此您可以选择一个合理的尺寸。 Array并且ListBuffer也是一个可变的数据类型,它又来了您的功能性风格的愿望。

更新:为了从Eratosthenes筛中获得良好的性能,我认为您需要将数据存储在本机数组中,这也违反了您对函数式编程风格的渴望。不过可能会有一个创造性的功能实现!

更新:哎呀!错过了! 如果您只用质数除以要测试的数的平方根作为除数, 这种方法也很好
!我错过了这一点,不幸的是,调整解决方案来做到这一点并不容易,因为我要向后存储素数。

更新:这是一个非常不实用的解决方案,至少只能检查平方根。

rnative,你可以使用一个ArrayVectorListBuffer对结果追加。Array但是,使用时,您需要估计要为其分配多少内存。幸运的是,我们知道它pi(n)大约等于,n / ln(n)因此您可以选择一个合理的尺寸。 Array并且ListBuffer也是一个可变的数据类型,它又来了您的功能性风格的愿望。

更新:为了从Eratosthenes筛中获得良好的性能,我认为您需要将数据存储在本机数组中,这也违反了您对函数式编程风格的渴望。不过可能会有一个创造性的功能实现!

更新:哎呀!错过了! 如果您只用质数除以要测试的数的平方根作为除数, 这种方法也很好
!我错过了这一点,不幸的是,调整解决方案来做到这一点并不容易,因为我要向后存储素数。

更新:这是一个非常不实用的解决方案,至少只能检查平方根。

import scala.collection.mutable.ListBuffer

def primesUnder(n: Int): List[Int] = {
  require(n >= 2)

  val primes = ListBuffer(2)
  for (i <- 3 to n) {
    if (prime(i, primes.iterator)) {
      primes += i
    }
  }

  primes.toList
}

// factors must be in sorted order
def prime(num: Int, factors: Iterator[Int]): Boolean = 
  factors.takeWhile(_ <= math.sqrt(num).toInt) forall(num % _ != 0)

或者我可以将Vectors与我的原始方法结合使用。 Vectors可能不是最佳解决方案,因为即使分期偿还O(1),他们也没有禁食的O(1)。



 类似资料:
  • 我必须找到小于或等于给定数字n的素数。我使用埃拉托斯特尼的筛子来寻找素数。 这是我写的代码: 这段代码适用于< code>integer值,但是我必须为< code>long数实现它。而Java不允许创建< code>long大小的数组,所以< code > boolean[]arr = new boolean[n 1];不起作用。有人能建议我解决这个问题的方法吗?

  • 问题内容: 我正在使用MAT比较两个堆转储。我每天都进行堆转储,并且每天增长约200兆。我认为该泄漏与java.util.zip有关,这是因为该表显示的内容,也是因为我们最近添加了一个新过程,该过程可以压缩和解压缩很多文件。(见图片) 此时,我打开支配器并过滤。 充气机 。那产生了大量的java.util.zip.Inflater。现在,我想看看是什么使它们保持打开状态,所以我选择了一个,并运行了

  • 类型错误:无法在导出处读取未定义的属性“user_id”.js user_code。(/user_code/node_modules/火库函数/库/云函数.js:112:27) 在下一个 (本机) 在 /user_code/node_modules/火基函数/库/云函数.js:28:71 在 __awaiter (/user_code/node_modules/火基函数/库/云函数.js:24:1

  • 本文向大家介绍Scala特质基础,包括了Scala特质基础的使用技巧和注意事项,需要的朋友参考一下 示例 这是Scala中特性的最基本版本。 由于没有为trait声明超类Identifiable,因此默认情况下它是从AnyRefclass扩展的。因为没有getIdentifier提供的定义Identifiable,所以Puppy该类必须实现它。然而,Puppy继承的实现printIdentific

  • 如何计算由像素组成的簇的质心? 我用它们的红色、蓝色和绿色值来表示像素(例如,)。 形心应该是集群中实例的平均值,但是如何计算像素之间的平均值呢? [我试图计算每种颜色的平均值(质心[“红色”]=总和\红色\像素\值/像素数等),但结果不正确] 如果有帮助的话,我用的是欧几里德距离函数。

  • 所以我尝试了这个在线挑战,但得到了运行时错误。请帮忙。我是编程新手。我附上了问题陈述和我的解决方案。 挑战使用Java语言,让函数取一个四位数的整数(至少有两位是不同的),并对该数字执行以下例程: 按降序和升序排列数字 注:在6174上表演例行公事总是会给你6174 (7641 - 1467 = 6174)。 例如:如果为则程序应返回:(通过1)5432-2345=3087,(通过2)8730-0

  • 本文向大家介绍在C ++中查找前N个质数的乘积,包括了在C ++中查找前N个质数的乘积的使用技巧和注意事项,需要的朋友参考一下 假设我们有一个数字n。我们必须找到1到n之间的质数的乘积。因此,如果n = 7,则输出将为210,因为2 * 3 * 5 * 7 = 210。 我们将使用Eratosthenes筛分法来查找所有素数。然后计算它们的乘积。 示例 输出结果