当前位置: 首页 > news >正文

(day18) leetcode 204.计数质数

描述

给定整数 n ,返回 所有小于非负整数 n 的质数的数量 。

示例 1:

输入:n = 10
输出:4
解释:小于 10 的质数一共有 4 个, 它们是 2, 3, 5, 7 。

示例 2:

输入:n = 0
输出:0

示例 3:

输入:n = 1
输出:0

提示:

  • 0 <= n <= 5 * 10^{6}

 超时代码

class Solution:def countPrimes(self, n: int) -> int:num=0for i in range(2,n):for j in range(2,int(math.sqrt(i))+1):if i%j == 0:breakelse:num+=1return num         

优化后的埃氏筛(
 来自Leecod题解powcai)

class Solution:  def countPrimes(self, n: int) -> int:  # 定义一个函数 countPrimes,接受一个整数 n 作为参数,并返回一个整数if n < 2:  # 如果 n 小于 2,直接返回 0,因为 2 是最小的素数,n 小于 2 不会有素数return 0  # 创建一个长度为 n 的列表 isPrimes,初始化为全 1,表示所有数都是潜在的素数isPrimes = [1] * n  # 将索引 0 和 1 对应的位置设为 0,因为 0 和 1 不是素数isPrimes[0] = isPrimes[1] = 0  # 遍历从 2 到 sqrt(n) 的整数(包含 sqrt(n)),这是埃拉托色尼筛法的优化步骤for i in range(2, int(math.sqrt(n)) + 1):  if isPrimes[i] == 1:  # 如果 i 是素数(即 isPrimes[i] 为 1)# 将 i 的所有倍数(从 i*i 开始,到 n 结束,步长为 i)标记为非素数(设为 0)isPrimes[i * i:n:i] = [0] * len(isPrimes[i * i:n:i])  # 返回 isPrimes 列表中所有素数的个数(即值为 1 的元素的个数)return sum(isPrimes)  

效率提升的关键在于埃拉托斯特尼筛法,简称埃式筛,也叫厄拉多塞筛法:

要得到自然数 n 以内的全部质数,必须把不大于 根号n 的所有质数的倍数剔除,剩下的就是质数。

基本思路

埃氏筛的基本思路是:

  1. 从2开始,依次标记每个素数的倍数为非素数。
  2. 重复上述步骤,直到处理完所有小于等于√n的数。

具体步骤

  1. 初始化

    • 创建一个布尔列表 isPrimes,长度为 n,初始化为全 TrueisPrimes[i] 表示数字 i 是否是素数。
    • isPrimes[0]isPrimes[1] 设为 False,因为0和1不是素数。
  2. 标记非素数

    • 2 开始遍历,每次找到一个未被标记为非素数的数字 i,将其所有倍数(从 i*i 开始)标记为非素数。
    • 之所以从 i*i 开始,是因为所有小于 i*i 的倍数在之前已经被处理过。
  3. 优化

    • 遍历的终止条件为 i <= √n,因为如果 i > √n,则 i 的所有倍数都已经在之前被标记。

优化思路的详细解释

为什么从 i*i 开始标记?

当你处理到某个数 i 时,所有比 i*i 小的 i 的倍数都已经在之前的步骤中被标记。例如,当处理 i = 3 时,3 的倍数 69 等已经在处理 i = 2 时被标记了。因此,从 i*i 开始标记可以避免重复操作,提高效率。

为什么只需处理到 √n

对于一个合数 n,它必然可以表示为两个整数的乘积,即 n = a * b。如果 ab 都大于 √n,则 a * b > n,这不可能。因此,在 √n 之后,只需要考虑标记较小因子的倍数。

 

相关文章:

  • 如何在idea安装git,使用gitee?
  • Pip换源:加速Python包安装的神操作,你get了吗?
  • Python与自动化脚本编写
  • 7.16做题总结
  • 昇思25天学习打卡营第19天|基于MobileNetv2的垃圾分类
  • LabVIEW阀门运动PCT测试
  • Knife4j的原理及应用详解(五)
  • [图解]SysML和EA建模住宅安全系统-14-黑盒系统规约
  • Python爬虫速成之路(2):爬天气情况
  • 机器学习——决策树(笔记)
  • 13--memcache与redis
  • 配置Redis时yml的格式导致报错
  • PostgreSQL 中如何处理数据的并发读写和锁等待超时?
  • dxf数据结构
  • linux的学习(四):磁盘,进程,定时,软件包的相关命令
  • hexo+github搭建个人博客
  • JS 中的深拷贝与浅拷贝
  • $translatePartialLoader加载失败及解决方式
  • 77. Combinations
  • iOS高仿微信项目、阴影圆角渐变色效果、卡片动画、波浪动画、路由框架等源码...
  • JavaScript工作原理(五):深入了解WebSockets,HTTP/2和SSE,以及如何选择
  • Java应用性能调优
  • js面向对象
  • Protobuf3语言指南
  • React+TypeScript入门
  • ReactNativeweexDeviceOne对比
  • TiDB 源码阅读系列文章(十)Chunk 和执行框架简介
  • 前端路由实现-history
  • 前端每日实战 2018 年 7 月份项目汇总(共 29 个项目)
  • 融云开发漫谈:你是否了解Go语言并发编程的第一要义?
  • 如何优雅地使用 Sublime Text
  • 设计模式走一遍---观察者模式
  • 优秀架构师必须掌握的架构思维
  • 正则与JS中的正则
  • #基础#使用Jupyter进行Notebook的转换 .ipynb文件导出为.md文件
  • (c语言)strcpy函数用法
  • (超详细)2-YOLOV5改进-添加SimAM注意力机制
  • (附源码)springboot 个人网页的网站 毕业设计031623
  • (含react-draggable库以及相关BUG如何解决)固定在左上方某盒子内(如按钮)添加可拖动功能,使用react hook语法实现
  • (汇总)os模块以及shutil模块对文件的操作
  • (生成器)yield与(迭代器)generator
  • (十)Flink Table API 和 SQL 基本概念
  • (转)使用VMware vSphere标准交换机设置网络连接
  • .NET C# 使用 iText 生成PDF
  • .net core IResultFilter 的 OnResultExecuted和OnResultExecuting的区别
  • .Net Remoting(分离服务程序实现) - Part.3
  • .NET WPF 抖动动画
  • .NET 线程 Thread 进程 Process、线程池 pool、Invoke、begininvoke、异步回调
  • .NET 自定义中间件 判断是否存在 AllowAnonymousAttribute 特性 来判断是否需要身份验证
  • .net的socket示例
  • 。。。。。
  • /proc/vmstat 详解
  • [ C++ ] STL_vector -- 迭代器失效问题
  • [AS3]URLLoader+URLRequest+JPGEncoder实现BitmapData图片数据保存
  • [BeginCTF]真龙之力