算法 - 質數(素數)

package main

import "time"
import "os"
import "fmt"
import "math"

func main() {
	timeStart := time.Now()
	// 獲取50萬之內的質數(普通算法,個數41538,耗時8.7秒)
	prime := GetPrimeLimit1(500000)
	print("個數", len(prime), ",")
	ShowTimeUsed(timeStart)
	// ShowPrime(prime)               // 顯示結果
	// SavePrime("prime1.txt", prime) // 保存結果

	timeStart = time.Now()
	// 獲取前5萬個質數(普通算法,個數50000,耗時12.6秒)
	prime = GetPrimeCount1(50000)
	print("個數", len(prime), ",")
	ShowTimeUsed(timeStart)
	// ShowPrime(prime)               // 顯示結果
	// SavePrime("prime2.txt", prime) // 保存結果

	timeStart = time.Now()
	// 獲取1億之內的質數(篩選算法,個數5761455,耗時1.4秒)
	prime = GetPrimeLimit2(100000000)
	print("個數", len(prime), ",")
	ShowTimeUsed(timeStart)
	// ShowPrime(prime)               // 顯示結果
	// SavePrime("prime1.txt", prime) // 保存結果

	timeStart = time.Now()
	// 判斷一個100億億級的大數是不是質數(耗時11.7秒)
	print(IsPrime(9223372036854775783), ",")
	ShowTimeUsed(timeStart)
}

// 獲取指定範圍內的質數(普通算法)
func GetPrimeLimit1(limit int) []int {
	// 先處理2,再處理2以上的質數
	prime := []int{2}
intLoop:
	// 2以上的質數都必須是奇數
	for i := 3; i <= limit; i += 2 {
		// 與以前的全部質數(除了2)相除,能整除則不是質數
		q := int(math.Sqrt(float64(i)))
		for j := 1; j < len(prime); j++ {
			if prime[j] <= q && i%prime[j] == 0 {
				continue intLoop
			}
		}
		prime = append(prime, i)
	}
	return prime
}

// 獲取指定數量的質數(普通算法)
func GetPrimeCount1(count int) []int {
	// 先處理2,再處理2以上的質數
	prime := []int{2}
intLoop:
	// 2以上的質數都必須是奇數
	for i := 3; ; i += 2 {
		// 與以前的全部質數(除了2)相除,能整除則不是質數
		q := int(math.Sqrt(float64(i)))
		for j := 1; j < len(prime); j++ {
			if prime[j] <= q && i%prime[j] == 0 {
				continue intLoop
			}
		}
		prime = append(prime, i)
		if len(prime) >= count {
			break
		}
	}
	return prime
}

// 獲取指定範圍內的質數(篩選算法)
// 此方法轉自http://blog.csdn.net/liukehua123/article/details/5482854
func GetPrimeLimit2(num int) []int {
	primeFlag := make([]bool, num, num)
	primeFlag[2] = true
	for i := 3; i < num; i += 2 {
		primeFlag[i] = true
	}
	for i := 3; i <= int(math.Sqrt(float64(num))); i++ {
		if primeFlag[i] {
			for j := i + i; j < num; j += i {
				primeFlag[j] = false
			}
		}
	}
	prime := []int{}
	for i := 0; i < num; i++ {
		if primeFlag[i] {
			prime = append(prime, i)
		}
	}
	return prime
}

// 判斷一個數是不是質數
// 此方法轉自http://blog.csdn.net/l04205613/article/details/6025118
func IsPrime(num int) bool {
	p := []int{4, 2, 4, 2, 4, 6, 2, 6}
	var j, q int
	i := 7
	if num == 1 {
		return false
	}
	if num == 2 || num == 3 || num == 5 {
		return true
	}
	if num%2 == 0 || num%3 == 0 || num%5 == 0 {
		return false
	}
	q = int(math.Sqrt(float64(num)))
	for i <= q {
		for j = 0; j < 8; j++ {
			if num%i == 0 {
				return false
			}
			i += p[j]
		}
		if num%i == 0 {
			return false
		}
	}
	return true
}

// 顯示函數執行時間
func ShowTimeUsed(timeStart time.Time) {
	println("耗時:", time.Now().Sub(timeStart).String())
}

// 顯示結果
func ShowPrime(prime []int) {
	for index, value := range prime {
		if (index+1)%10 == 0 { // 每10個質數一行
			println(value)
		} else if index+1 == len(prime) { // 最後一個質數後面不添加逗號
			print(value)
		} else {
			print(value, ", ")
		}
	}
	println()
}

// 保存結果
func SavePrime(filename string, prime []int) {
	f, err := os.OpenFile(filename, os.O_CREATE|os.O_TRUNC|os.O_RDWR, 0660)
	if err != nil {
		println("文件打開錯誤")
		return
	}
	defer f.Close()

	for index, value := range prime {
		if (index+1)%10 == 0 { // 每10個質數一行
			f.WriteString(fmt.Sprintf("%d\n", value))
		} else if index+1 == len(prime) { // 最後一個質數後面不添加逗號
			f.WriteString(fmt.Sprintf("%d", value))
		} else {
			f.WriteString(fmt.Sprintf("%d, ", value))
		}
	}
}
相關文章
相關標籤/搜索