GoLang 中的随机数的示例代码

作者:根号三 时间:2024-04-26 17:16:51 

随机数我们都知道,就是计算机通过某种算法,“随机”的生成一个数字。很多编程语言都有内置的方法来生成随机数,那么 GoLang 中是怎样一种情况呢?

伪随机数

我们都知道“随机数”在现实生活中的概念,可能你随手抛一个硬币,就可以说其结果是随机的,但是在计算机中要确定一个“随机数”真的是“随机数”,那可是有标准的,不是你随随便便说是就是。

根据密码学原理,要想对一个“随机数”进行随机性检验有以下几个标准:

  1. 统计学伪随机性 - 在给定的随机比特流样本中,1 的数量大致等于 0 的数量,也就是说,“10”“01”“00”“11” 四者数量大致相等。说人话就是:“一眼看上去是随机的”。

  2. 密码学安全伪随机性 - 就是给定随机样本的一部分和随机算法,不能有效的演算出随机样本的剩余部分。

  3. 真随机性 - 其定义为随机样本不可重现。

根据以上几个标准,其对应的随机数也就分为以下几类:

  1. 伪随机数 - 满足第一个条件的随机数。

  2. 密码学安全的伪随机数 - 同时满足前两个条件的随机数。可以通过密码学安全伪随机数生成器计算得出。

  3. 真随机数 -同时满足三个条件的随机数。

了解了以上几个概念,我们就知道了“伪随机数”其实就是一个“看似随机,实则并不真正随机”的数字。

伪随机数生成器

在实际应用中大部分情况下伪随机数就足够了。这些数列是“似乎”随机的数,实际上它们是通过一个固定的、可以重复的计算方法产生的。因为它们实际上是可以计算出来的,所以它们并不真正地随机,但是它们具有类似于随机数的统计特征。产生这样的结果的生成器我们叫做伪随机数生成器。

一般只有在密码学场景中,我们才需要使用“真随机数”。

在大部分编程语言中,提供的都是“伪随机数生成器”,例如 JS 中的 Math.random() , GoLang 中的 math/rand 包。

GoLang 中的伪随机数

在 GoLang 中,我们可以通过 math/rand 包里的方法来生成一个伪随机数:


package main
import (
"fmt"
"math/rand"
)
func main() {
fmt.Println(rand.Int())  // => 134020434
}

上面的代码中,我们通过 rand.Int() 方法来生成一个伪随机数。看起来好像没什么问题嘛,人家也很 OK 啦。

但是细心的你会发现,你在自己电脑上运行上面的代码竟然和我的一样。无论你怎么运行,它都一样。

我们知道 JS 中的 Math.random() 每次都会返回一个不一样的数字,但是 GoLang 中的伪随机数生成器默认情况下竟然会返回相同的数值,这还不反了天了?

都是伪随机数生成器,为什么差别就这么大呢?这里我们就要了解一下“随机种子”的概念啦。

随机种子

我们知道,伪随机数,是使用一个确定性的算法计算出来的似乎是随机的数序,因此伪随机数实际上并不随机。

那么自然,在计算伪随机数时假如使用的开始值不变的话,那么算法计算出的伪随机数的数序自然也是不变的咯。

这个“开始值”,就被称为随机种子。

查阅文档,我们得知, Int() 函数是从 default Source (默认源)中产生的伪随机数。

而这个 default Source ,我们从Seed 部分可以看到,如果你没有设置随机种子,那么默认初始种子总是从 1 开始。

既然随机种子一样,那自然其结果也是一样的。

随机的伪随机数

我们已经知道了默认随机种子是从 1 开始,那么我们只要在每次生成随机数之前先设置一个不一样的种子,那么其结果自然也就不一样了。

我们要尽可能保证每次伪随机数生成器工作时使用的是不同的种子,通常的做法是采用当前时间作为种子。


package main
import (
"fmt"
"math/rand"
"time"
)
func main() {
rand.Seed(int64(time.Now().UnixNano()))
fmt.Println(rand.Int())
}

这样,由于种子不同,我们每次运行的结果也就不一样。我们就能达到获取伪随机数的目的啦。

真随机数

如果我们的应用对安全性要求比较高,需要使用真随机数的话,那么可以使用 crypto/rand 包中的方法。


package main
import (
"crypto/rand"
"fmt"
"math/big"
)
func main() {
// 生成 20 个 [0, 100) 范围的真随机数。
for i := 0; i < 20; i++ {
 result, _ := rand.Int(rand.Reader, big.NewInt(100))
 fmt.Println(result)
}
}

上面的程序每次运行的结果都是不一样的,会真正随机的生成随机数。

访问:https://github.com/sqrthree/sqrthree.github.io/issues

来源:https://juejin.im/post/5ab0c7f06fb9a028d936fe5c

标签:GoLang,随机数
0
投稿

猜你喜欢

  • asp如何在线压缩Access数据库?

    2010-06-26 12:25:00
  • MySQL中使用MD5加密的实现

    2024-01-15 14:11:38
  • vue生成随机验证码的示例代码

    2023-07-02 16:57:09
  • MYSQL中varchar和TEXT的相关问题详析

    2024-01-23 00:58:42
  • 英文罚抄引发的艺术创意

    2008-05-13 12:02:00
  • 关于web标准的思考

    2008-10-05 10:07:00
  • 用Python编写简单的微博爬虫

    2022-10-07 04:58:34
  • mysql仿oracle的decode效果查询

    2024-01-12 22:04:00
  • 用Python实现BP神经网络(附代码)

    2023-11-24 17:20:11
  • Python实现求解最大公约数的五种方法总结

    2023-07-02 20:34:46
  • Python 实现定积分与二重定积分的操作

    2023-08-05 18:22:27
  • 整理及优化CSS代码的七个原则[译]

    2009-04-23 12:35:00
  • Symfony2实现从数据库获取数据的方法小结

    2023-11-14 13:56:35
  • 详解如何在Go项目中输出版本信息

    2024-04-26 17:17:20
  • Python3实现打格点算法的GPU加速实例详解

    2022-05-26 23:20:30
  • Python中高阶函数的小实践分享

    2022-05-24 11:59:04
  • [组图]手把手教你制作ASP留言本

    2007-09-22 09:32:00
  • 78行Python代码实现现微信撤回消息功能

    2021-12-17 17:41:19
  • 在DOS界面如何运行python的py文件

    2022-05-01 16:39:33
  • JavaScript Length 字符长度函数

    2008-12-12 12:29:00
  • asp之家 网络编程 m.aspxhome.com