golang学习笔记 ---rand

在Golang中,有两个包提供了rand,分别为 "math/rand" 和 "crypto/rand",  对应两种应用场景。

  • "math/rand" 包实现了伪随机数生成器。也就是生成 整形和浮点型。 该包中根据生成伪随机数是是否有种子(可以理解为初始化伪随机数),可以分为两类:
    • 有种子。通常以时钟,输入输出等特殊节点作为参数,初始化。该类型生成的随机数相比无种子时重复概率较低。
    • 无种子。可以理解为此时种子为1, Seek(1)

golang 随机数有一个很有趣的地方,如果我们不自行定义随机数种子的话,每次生成的随机数都是一样的。golang 在默认随机的时候,固定以数字 1 作为种子。既然种子都固定了的话,那每次执行的结果当然是一样的。比如 第二个for 循环,打印 10 个随机数,每次运行结果是一模一样的。 

要解决这个问题,就需要以时间作为随机数种子。如第一个for循环:

package main

import (
	"fmt"
	"math/rand"
	"time"
)

func main() {
      //以时间为随机种子 for i := 0; i < 10; i++ r := rand.New(rand.NewSource(time.Now().UnixNano())) fmt.Printf("%d ", r.Int31()) } fmt.Println(" ")
    //无种子,其实是固定以1为种子 for i := 0; i < 10; i++ { fmt.Printf("%d ", rand.Int31()) } }

  输出:

第一次运行结果

276310213 276310213 276310213 939632684 939632684 939632684 939632684 939632684 939632684 939632684

1298498081 2019727887 1427131847 939984059 911902081 1474941318 140954425 336122540 208240456 646203300

第二次运行结果

1101837366 1101837366 1101837366 1101837366 1101837366 1101837366 1101837366 1101837366 1101837366 1101837366

1298498081 2019727887 1427131847 939984059 911902081 1474941318 140954425 336122540 208240456 646203300

第三次运行结果

483465292 483465292 483465292 483465292 1406725908 1406725908 1406725908 1406725908 1406725908 1406725908

1298498081 2019727887 1427131847 939984059 911902081 1474941318 140954425 336122540 208240456 646203300

 

常用的方法有:(以有种子的为例,无种子的直接通过 rand 调用对应的方法)

1> 按类型随机类:
func (r *Rand) Int() int
func (r *Rand) Int31() int32
func (r *Rand) Int63() int64
func (r *Rand) Uint32() uint32
func (r *Rand) Float32() float32  // 返回一个取值范围在[0.0, 1.0)的伪随机float32值
func (r *Rand) Float64() float64  // 返回一个取值范围在[0.0, 1.0)的伪随机float64值

2>指定随机范围类:
func (r *Rand) Intn(n int) int
func (r *Rand) Int31n(n int32) int32
func (r *Rand) Int63n(n int64) int64

 

拓展:对于需要随机指定位数的,当位数不够是,可以通过前边补0达到长度一致,如

package main
import (
    "fmt"
    "math/rand"
)
func main() {
    // 随机产生6位长度伪随机数
    for i := 0; i < 10; i++ {
        fmt.Printf("%.6d ", rand.Int31()%1000000)
    }
}

  输出

498081 727887 131847 984059 902081 941318 954425 122540 240456 203300

 

package main
import (
    "fmt"
    "math/rand"
    "time"
)
func init() {
    //以时间作为初始化种子
    rand.Seed(time.Now().UnixNano())
}
func main() {
    a := rand.Int()
    fmt.Println(a)
    b := rand.Intn(100)
    fmt.Println(b)
    c := rand.Float32()
    fmt.Println(c)
}

  输出:

1910927962
45
0.9863281

  

”crypto/rand“ 包实现了用于加解密的更安全的随机数生成器:该包中常用的是 func Read(b []byte) (n int, err error) 这个方法, 将随机的byte值填充到b 数组中,以供b使用。示例如下:

package main

import (
	"crypto/rand"
	"fmt"
)

func main() {
	b := make([]byte, 20)
	fmt.Println(b) //

	_, err := rand.Read(b)
	if err != nil {
		fmt.Println(err.Error())
	}

	fmt.Println(b)
}

几点注意项:

1、如果不使用rand.Seed(seed int64),每次运行,得到的随机数会一样,程序不停止,一直获取的随机数是不一样的;

2、每次运行时rand.Seed(seed int64),seed的值要不一样,这样生成的随机数才会和上次运行时生成的随机数不一样;

3、rand.Intn(n int)得到的随机数int i,0 <= i < n。