使用 Go 语言统计 0-200000 的数字中,哪些是素数?

题目

使用 Go 语言统计 0-200000的数字中,哪些是素数?

思路

两种方法:

  1. 单循环遍历 1-200000 数字,并判断是否是素数。

  2. 使用了 Goroutine 和通道实现并发:

    • 通过创建两个通道 intChanprimeChan,以及一个 sync.WaitGroup 来协调 Goroutine 的执行。

    • 然后,启动三个 Goroutine:

      • InputIntChan:向 intChan 通道中写入整数数据。
      • PrimeCompute:从 intChan 通道中读取整数数据,并判断是否为素数,将素数写入 primeChan 通道。
      • readPrimeChan:从 primeChan 通道中读取素数数据并输出。

      最后,使用 sync.WaitGroup 等待所有 Goroutine 执行完毕。

代码实现

单循环实现

go 复制代码
package main

import (
	"fmt"
	"math"
	"time"
)

func main() {
	start := time.Now().Unix() // 记录开始时间
	primeChan := make(chan int, 2000000) // 创建一个缓冲大小为2000000的素数通道
	for i := 0; i <= 200000; i++ { // 判断从0到200000的整数是否为素数
		if isPrime(i) { // 如果是素数
			primeChan <- i // 写入素数通道
		}
	}
	end := time.Now().Unix() // 记录结束时间

	fmt.Println("总耗时时间:", end-start) // 输出总耗时时间
}

// 判断一个整数是否为素数
func isPrime(n int) bool {
	if n <= 1 {
		return false
	}

	for i := 2; i <= int(math.Sqrt(float64(n))); i++ { // 只需遍历到sqrt(n)即可,减少计算量
		if n%i == 0 {
			return false
		}
	}

	return true
}

Goroutine 和通道实现

好的,以下是给代码加上注释后的版本:

go 复制代码
package main

import (
	"fmt"
	"math"
	"strconv"
	"sync"
)

func main() {

	var wg sync.WaitGroup
	wg.Add(3)
	intChan := make(chan int, 1000)      // 创建一个缓冲大小为1000的整数型通道
	primeChan := make(chan int, 2000000) // 创建一个缓冲大小为2000000的整数型通道

	go InputIntChan(intChan, &wg) // 启动写入整数的协程

	go PrimeCompute(intChan, primeChan, &wg) // 启动计算素数并写入素数通道的协程

	go readIntChan(primeChan, &wg) // 启动读取素数并输出的协程

	wg.Wait() // 等待所有协程结束
}

// 判断一个整数是否为素数
func isPrime(n int) bool {
	if n <= 1 {
		return false
	}

	for i := 2; i <= int(math.Sqrt(float64(n))); i++ { // 只需遍历到sqrt(n)即可,减少计算量
		if n%i == 0 {
			return false
		}
	}

	return true
}

// 往通道中写入整数
func InputIntChan(intChan chan<- int, wg *sync.WaitGroup) {
	for i := 0; i <= 200000; i++ { // 写入200000个整数
		intChan <- i // 写入整数通道
		fmt.Println("写入Int通道==" + strconv.Itoa(i))
	}
	close(intChan) // 写入完成,关闭通道
	wg.Done()      // 协程结束,减少计数器
}

// 判断整数是否为素数,并往素数通道中写入素数
func PrimeCompute(intChan <-chan int, primeChan chan<- int, wg *sync.WaitGroup) {
	for i := range intChan { // 循环判断每个整数是否为素数
		if isPrime(i) {
			primeChan <- i // 如果是素数,往素数通道中写入素数
		}
	}
	close(primeChan) // 计算完毕,关闭素数通道
	wg.Done()        // 协程结束,减少计数器
}

// 从素数通道中读取素数并输出
func readIntChan(primeChan <-chan int, wg *sync.WaitGroup) {
	for i := range primeChan { // 循环读取素数通道中的素数
		fmt.Println("从通道中读取素数==" + strconv.Itoa(i))
	}
	wg.Done() // 协程结束,减少计数器
}

优化思路

创造多个协程处理写入读取通道

相关推荐
loser~曹4 分钟前
基于快速排序解决 leetcode hot215 查找数组中第k大的数字
数据结构·算法·leetcode
啊阿狸不会拉杆8 分钟前
第二十一章:Python-Plotly库实现数据动态可视化
开发语言·python·plotly
Dream it possible!10 分钟前
LeetCode 热题 100_打家劫舍(83_198_中等_C++)(动态规划)
c++·算法·leetcode·动态规划
zhouziyi070115 分钟前
【蓝桥杯14天冲刺课题单】Day 8
c++·算法·蓝桥杯
滴答滴答嗒嗒滴18 分钟前
Python小练习系列 Vol.12:学生信息排序(sorted + key函数)
开发语言·python
SylviaW0819 分钟前
python-leetcode 62.搜索插入位置
数据结构·算法·leetcode
愚润求学1 小时前
【C++】vector常用方法总结
开发语言·c++·vector
赴前尘1 小时前
Go+Gin实现安全多文件上传:带MD5校验的完整解决方案
安全·golang·gin
天天进步20151 小时前
Python项目-基于Flask的个人博客系统设计与实现(1)
开发语言·python·flask
安然无虞1 小时前
31天Python入门——第20天:魔法方法详解
开发语言·后端·爬虫·python