0

0

Go语言中高效生成素数:Atkin筛法详解

碧海醫心

碧海醫心

发布时间:2025-11-24 19:53:42

|

393人浏览过

|

来源于php中文网

原创

Go语言中高效生成素数:Atkin筛法详解

本文深入探讨了在go语言中高效生成素数的方法,纠正了对素数判断的常见误解,并详细介绍了优化的atkin筛法。通过提供完整的go语言实现代码,文章解析了atkin筛法的核心原理,包括基于二次形式的素数筛选逻辑和优化条件,旨在帮助开发者理解并应用先进算法来生成指定范围内的素数。

素数及其识别挑战

素数是大于1的自然数,除了1和它本身以外不再有其他因数。例如,2、3、5、7都是素数。在编程中,一个常见的误解是尝试通过 i % i == 0 && i % 1 == 0 这样的条件来识别素数。然而,这个条件对于任何整数 i(只要 i 不为零)都成立,因为它仅仅表达了任何数都能被自身和1整除的基本数学事实,而未能区分素数与合数。因此,要准确地生成或判断素数,我们需要依赖更为复杂的算法。

素数生成算法概述

生成指定上限内的所有素数通常需要使用“筛法”(Sieve method)。其中最古老且知名的是埃拉托斯特尼筛法 (Sieve of Eratosthenes)。它通过从2开始,逐个标记合数的倍数来筛选出素数。虽然埃拉托斯特尼筛法简单易懂,但对于非常大的上限,其效率会受到限制。

为了进一步优化素数生成过程,数学家们开发了更高效的算法,例如Atkin筛法 (Sieve of Atkin)。Atkin筛法是埃拉托斯特尼筛法的一个优化变体,它利用了二次形式的数学性质来减少标记操作的次数,从而在理论上提供更好的时间复杂度,尤其是在处理大规模素数生成时。

Atkin筛法原理

Atkin筛法的核心思想是利用特定的二次方程来识别潜在的素数,然后通过一个后续步骤来排除合数。它主要基于以下三个二次形式:

立即学习go语言免费学习笔记(深入)”;

  1. 4x^2 + y^2
  2. 3x^2 + y^2
  3. 3x^2 - y^2

这些形式在满足特定模运算条件时,可以帮助我们初步确定一个数是否为素数。具体来说,Atkin筛法会迭代 x 和 y 的值,计算上述二次形式的结果 n。如果 n 小于等于上限 N 且满足特定的模12条件,则将 n 的素数状态(通常用布尔值表示)进行翻转。

模12条件:

  • 如果 n = 4x^2 + y^2 且 n % 12 == 1 或 n % 12 == 5,则 n 可能是素数。
  • 如果 n = 3x^2 + y^2 且 n % 12 == 7,则 n 可能是素数。
  • 如果 n = 3x^2 - y^2 且 n % 12 == 11 (且 x > y),则 n 可能是素数。

在所有可能的 x 和 y 组合处理完毕后,Atkin筛法会进行第二阶段的筛选:遍历已经标记为潜在素数的数 n。如果 n 是一个素数,则所有 n^2 的倍数(n^2, 2n^2, 3n^2...)都是合数,需要将它们标记为非素数。最后,将2和3这两个特殊素数明确标记,并收集所有最终被标记为素数的数字。

喵记多
喵记多

喵记多 - 自带助理的 AI 笔记

下载

Go语言实现

下面是Atkin筛法在Go语言中的一个完整实现,用于生成指定上限 N 内的所有素数:

package main

import (
    "fmt"
    "math"
)

// N 定义了生成素数的上限
const N = 100

func main() {
    var x, y, n int
    // 计算N的平方根,用于优化循环边界
    nsqrt := math.Sqrt(N)

    // is_prime 数组用于标记每个数字是否为素数。
    // 初始时所有元素默认为false,表示未知或非素数。
    // 经过第一阶段的二次形式计算,某些位置会被翻转为true,表示可能是素数。
    is_prime := make([]bool, N+1) // 数组大小为 N+1 以便索引到 N

    // 第一阶段:根据二次形式和模12条件翻转is_prime状态
    for x = 1; float64(x) <= nsqrt; x++ {
        for y = 1; float64(y) <= nsqrt; y++ {
            // 形式一: 4x^2 + y^2
            n = 4*(x*x) + y*y
            if n <= N && (n%12 == 1 || n%12 == 5) {
                is_prime[n] = !is_prime[n] // 翻转状态
            }

            // 形式二: 3x^2 + y^2
            n = 3*(x*x) + y*y
            if n <= N && n%12 == 7 {
                is_prime[n] = !is_prime[n] // 翻转状态
            }

            // 形式三: 3x^2 - y^2
            n = 3*(x*x) - y*y
            // 确保 x > y 是为了避免重复计算和负数结果,且满足Atkin算法的要求
            if x > y && n <= N && n%12 == 11 {
                is_prime[n] = !is_prime[n] // 翻转状态
            }
        }
    }

    // 第二阶段:排除合数
    // 遍历所有可能的素数(从5开始,因为2和3是特殊处理的),
    // 如果一个数 n 被标记为素数,则它的平方的倍数都是合数。
    for n = 5; float64(n) <= nsqrt; n++ {
        if is_prime[n] { // 如果 n 已经被标记为潜在素数
            // 将 n*n 的所有倍数标记为非素数
            // 注意:这里从 n*n 开始,因为小于 n*n 的合数应该已经被更小的素数处理过了
            for y = n * n; y <= N; y += n * n {
                is_prime[y] = false
            }
        }
    }

    // 明确标记2和3为素数,因为它们不符合上述二次形式的模12条件
    if N >= 2 {
        is_prime[2] = true
    }
    if N >= 3 {
        is_prime[3] = true
    }

    // 收集所有素数到一个切片中
    primes := make([]int, 0, N/math.Log(float64(N))) // 预估素数数量以优化容量
    for x = 0; x <= N; x++ {
        if is_prime[x] {
            primes = append(primes, x)
        }
    }

    // 打印生成的素数
    fmt.Printf("Primes up to %d:\n", N)
    for _, p := range primes {
        fmt.Println(p)
    }
}

代码解析

  1. 初始化 (const N, nsqrt, is_prime):

    • N 定义了我们想要生成素数的上限。
    • nsqrt 是 N 的平方根,它用于优化循环的边界,因为很多操作只需要迭代到 sqrt(N)。
    • is_prime 是一个布尔型切片(在示例中为数组),其索引代表数字,值为 true 表示该数字是素数,false 表示不是。初始时所有值都为 false。
  2. 第一阶段:二次形式筛选:

    • 两个嵌套循环遍历 x 和 y,范围从1到 nsqrt。
    • 在循环内部,计算三个二次形式 4x^2 + y^2、3x^2 + y^2 和 3x^2 - y^2 的结果 n。
    • 对于每个 n,如果它在有效范围内 (n
  3. 第二阶段:排除合数:

    • 这个循环从 n = 5 开始,到 nsqrt 结束。
    • 如果 is_prime[n] 为 true(表示 n 经过第一阶段后被认为是潜在素数),那么 n 确实是一个素数。
    • 接着,将 n 的平方 n*n 及其所有倍数(n*n + n*n,n*n + 2*n*n 等)在 is_prime 数组中标记为 false,因为这些都是合数。
  4. 特殊素数处理:

    • 素数2和3不符合Atkin筛法第一阶段的模12条件,因此需要单独将 is_prime[2] 和 is_prime[3] 设为 true。
  5. 收集和打印:

    • 最后,遍历 is_prime 数组,将所有标记为 true 的索引(即素数)收集到一个 primes 切片中,并打印出来。

注意事项与优化

  • 内存使用: Atkin筛法需要一个与上限 N 大小相同的布尔数组,因此当 N 非常大时,内存消耗会成为一个考虑因素。
  • 性能: Atkin筛法的理论时间复杂度优于埃拉托斯特尼筛法,尤其是在 N 趋于无穷大时。它避免了埃拉托斯特尼筛法中对所有倍数的重复标记,而是利用更复杂的数学性质来减少操作。
  • 并发性: 对于极大的 N,可以考虑将Atkin筛法的某些阶段并行化,例如将 x 和 y 的迭代范围分配给不同的goroutine处理,以进一步提高性能。然而,这会增加代码的复杂性,并且需要谨慎处理共享内存的并发访问
  • 上限选择: 示例代码中的 N = 100 仅用于演示。在实际应用中,可以根据需求设置更大的上限。

总结

生成素数是计算机科学中的一个经典问题,高效的素数生成算法在密码学、数论研究等领域都有广泛应用。Atkin筛法提供了一种优化的解决方案,通过利用二次形式和模运算,显著提高了生成大规模素数的效率。理解并掌握其Go语言实现,能够帮助开发者在需要素数列表的场景中,选择并应用更为专业的算法。

相关专题

更多
c语言const用法
c语言const用法

const是关键字,可以用于声明常量、函数参数中的const修饰符、const修饰函数返回值、const修饰指针。详细介绍:1、声明常量,const关键字可用于声明常量,常量的值在程序运行期间不可修改,常量可以是基本数据类型,如整数、浮点数、字符等,也可是自定义的数据类型;2、函数参数中的const修饰符,const关键字可用于函数的参数中,表示该参数在函数内部不可修改等等。

523

2023.09.20

c语言const用法
c语言const用法

const是关键字,可以用于声明常量、函数参数中的const修饰符、const修饰函数返回值、const修饰指针。详细介绍:1、声明常量,const关键字可用于声明常量,常量的值在程序运行期间不可修改,常量可以是基本数据类型,如整数、浮点数、字符等,也可是自定义的数据类型;2、函数参数中的const修饰符,const关键字可用于函数的参数中,表示该参数在函数内部不可修改等等。

523

2023.09.20

Go中Type关键字的用法
Go中Type关键字的用法

Go中Type关键字的用法有定义新的类型别名或者创建新的结构体类型。本专题为大家提供Go相关的文章、下载、课程内容,供大家免费下载体验。

233

2023.09.06

go怎么实现链表
go怎么实现链表

go通过定义一个节点结构体、定义一个链表结构体、定义一些方法来操作链表、实现一个方法来删除链表中的一个节点和实现一个方法来打印链表中的所有节点的方法实现链表。

444

2023.09.25

go语言编程软件有哪些
go语言编程软件有哪些

go语言编程软件有Go编译器、Go开发环境、Go包管理器、Go测试框架、Go文档生成器、Go代码质量工具和Go性能分析工具等。本专题为大家提供go语言相关的文章、下载、课程内容,供大家免费下载体验。

246

2023.10.13

0基础如何学go语言
0基础如何学go语言

0基础学习Go语言需要分阶段进行,从基础知识到实践项目,逐步深入。php中文网给大家带来了go语言相关的教程以及文章,欢迎大家前来学习。

693

2023.10.26

Go语言实现运算符重载有哪些方法
Go语言实现运算符重载有哪些方法

Go语言不支持运算符重载,但可以通过一些方法来模拟运算符重载的效果。使用函数重载来模拟运算符重载,可以为不同的类型定义不同的函数,以实现类似运算符重载的效果,通过函数重载,可以为不同的类型实现不同的操作。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

191

2024.02.23

Go语言中的运算符有哪些
Go语言中的运算符有哪些

Go语言中的运算符有:1、加法运算符;2、减法运算符;3、乘法运算符;4、除法运算符;5、取余运算符;6、比较运算符;7、位运算符;8、按位与运算符;9、按位或运算符;10、按位异或运算符等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

228

2024.02.23

Golang gRPC 服务开发与Protobuf实战
Golang gRPC 服务开发与Protobuf实战

本专题系统讲解 Golang 在 gRPC 服务开发中的完整实践,涵盖 Protobuf 定义与代码生成、gRPC 服务端与客户端实现、流式 RPC(Unary/Server/Client/Bidirectional)、错误处理、拦截器、中间件以及与 HTTP/REST 的对接方案。通过实际案例,帮助学习者掌握 使用 Go 构建高性能、强类型、可扩展的 RPC 服务体系,适用于微服务与内部系统通信场景。

8

2026.01.15

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Go 教程
Go 教程

共32课时 | 3.8万人学习

Go语言实战之 GraphQL
Go语言实战之 GraphQL

共10课时 | 0.8万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号