当前位置: 首页 > news >正文

掷骰子式的乐趣:探究C语言生成随机数的奥秘

掷骰子式的乐趣:探究C语言生成随机数的奥秘

    • 一、引言
    • 二、C标准库的rand函数
    • 三、srand函数的使用
    • 四、基于时间的种子生成
    • 五、高质量随机数的应用

一、引言

C语言中生成随机数是一项非常重要的功能,因为许多现代应用程序需要使用随机数。随机数可以用于密码学、随机化算法、统计分析、模拟等多种场景。以下是一些应用场景的例子:

  1. 密码学:随机数在密码学中被广泛使用,例如生成随机密钥或者初始化向量。随机数可以使加密的结果更加难以预测,从而更加安全。
  2. 随机化算法:随机数常常用于各种算法中来实现随机化选择。例如在排序算法中,随机数可以用于打乱数组中的元素来使得排序更加随机。
  3. 统计分析:随机数可以用于模拟和推断统计学模型。例如,模拟股票价格和天气预测需要使用随机数。
  4. 模拟:随机数也可以用于模拟实验,例如游戏、实验室实验等。在这些场景下,随机数可以帮助创建更加真实的情况,使得实验更加有说服力。

综上所述,C语言中生成随机数的重要性在于,它可以使得现代应用程序更加安全、高效、可靠。

二、C标准库的rand函数

rand是C语言中的一个随机数生成函数,用于生成一个伪随机数。其基本用法如下:

#include <stdlib.h>
int rand(void);

其中,需要引入stdlib.h头文件。

使用rand函数时,需要先通过srand函数设置种子值。种子值的设置可以是时间戳,也可以是其他的变量值。如果没有指定种子值,则会使用默认的种子值1。

另外,需要注意的是,rand函数返回的是一个整型随机数,其范围是0到RAND_MAX。其中,RAND_MAX是一个常量,表示随机数的最大值。在标准C中,RAND_MAX至少为32767。

rand函数的原理是:在计算机内部,随机数通过数学方法生成,具体来说是通过伪随机数发生器生成。在C语言中,伪随机数发生器会根据设置的种子值,计算出一串随机数序列。每次调用rand函数时,其实是从这个序列中取出一个随机值。因此,如果使用相同的种子值,每次生成的随机数序列都是一样的。

当我们需要在C语言程序中随机生成一些数字或选项时,可以使用rand函数。下面是一个使用rand函数生成随机数的示例程序:

#include <stdio.h>
#include <stdlib.h>
#include <time.h>int main()
{int i;srand(time(NULL));// 生成10个随机数for (i = 0; i < 10; i++) {printf("%d ", rand());}return 0;
}

在上面的程序中,我们使用了srand函数来设置种子,使得每次运行程序都可以产生不同的随机数。在本例中,我们使用了当前时间作为种子。接着,我们使用rand函数来生成随机数,并打印出来。这个程序会生成10个随机数。

三、srand函数的使用

在C语言中,srand函数用于设置随机数种子,以便让程序生成不同的随机数序列。其语法如下:

void srand(unsigned int seed);

seed参数是随机数种子,它可以是整数。在使用rand函数之前,我们需要先调用srand函数来设置种子。通常,我们使用当前时间作为种子,可以使用time函数来获取当前时间,代码如下:

srand(time(NULL));

注意,time函数需要引入头文件time.h。在某些情况下,我们可能需要使用固定的种子,这样可以产生确定的随机数序列,比如用于调试程序。在这种情况下,我们可以将种子设定为固定的值。

srand(1234); // 固定种子为1234

总体来说,我们应当尽可能使用随机的种子生成随机数,以保证每次运行程序生成不同的随机数。

在C语言中,常常使用rand函数生成随机数。但是,如果我们不加特别处理,使用rand函数生成的随机数可能存在某些问题,比如出现重复的随机数、周期性等问题。为了生成更加高质量的随机数,我们可以采用以下方法。

  1. 使用time函数获取当前时间作为随机数种子

我们可以使用time函数获取当前时间作为随机数种子。由于时间是不可预测的,因此每次运行程序都可以生成不同的随机数序列。

srand(time(NULL)); // 使用当前时间作为随机数种子
  1. 采用更好的随机数算法

rand函数使用的生成随机数的算法通常是线性同余法(LCG),这种算法生成的随机数序列可能存在某些问题,比如随机性不足、周期较短等。因此,我们可以选择更好的随机数生成算法,比如Xorshift算法、Mersenne Twister算法等。

  1. 生成更广泛的随机数

由于rand函数生成的是伪随机数,因此可能存在某些规律。为了生成更加随机的数,我们可以采用一些随机数生成库,比如librandom库、libgcrypt库等。这些库可以生成更加广泛的随机数,以满足更严格的随机性要求。

综上所述,为了生成更加高质量的随机数,我们可以采用更好的随机数算法,使用更广泛的随机数生成库,或者使用time函数获取当前时间作为随机数种子。

四、基于时间的种子生成

在C语言中,我们可以使用time函数获取当前时间,然后将时间作为随机数种子,生成更好的、更随机的随机数。time函数的原型如下:

time_t time(time_t *t);

其中,time_t类型表示时间的值,t是一个指向time_t类型对象的指针。我们可以将t设置为NULL,让time函数返回当前时间,如下所示:

srand(time(NULL));

在上面的代码中,time(NULL)返回当前时间的秒数,将其作为srand函数的种子,可以让程序每次运行时生成一个不同的、随机的随机数序列。需要注意的是,time函数返回的时间值是一个长整型,需要将其转换成unsigned int类型才能用作srand函数的种子,但是一般情况下可以直接将其传入srand函数中,因为时间值的范围一般大于unsigned int类型所能表示的范围,所以可以直接截取时间值的低位作为种子。

在C语言中,我们可以使用time函数获取当前时间,将当前时间作为随机数种子,然后使用rand函数生成随机数。下面是一个使用time函数生成随机数的示例程序。

#include <stdio.h>
#include <stdlib.h>
#include <time.h>int main()
{int i, r;srand(time(NULL));// 生成10个随机数for (i = 0; i < 10; i++) {r = rand() % 100; // 生成0~99之间的随机数printf("%d ", r);}return 0;
}

在上面的程序中,我们使用了srand函数和time函数来生成随机数。在调用srand函数时,传入的参数是time(NULL),它返回当前时间的秒数。然后,我们使用rand函数生成随机数,并将其取模,以生成0~99之间的随机数。最后,我们打印生成的10个随机数。由于使用了时间函数,每次运行程序都可以生成不同的随机数序列。

五、高质量随机数的应用

  1. 密码学

在密码学中,需要生成高质量的随机数来保证安全性,比如生成密钥、生成随机数序列等。下面是一个使用C语言中的openssl库来生成随机数的代码示例:

#include <openssl/rand.h>
#include <stdio.h>int main() {unsigned char random_bytes[16];RAND_bytes(random_bytes, 16);printf("随机数是:");for (int i = 0; i < 16; i++) {printf("%02x", random_bytes[i]);}printf("\n");return 0;
}
  1. 模拟

在模拟中,需要生成符合一定分布的随机数来模拟实际场景下的随机事件,比如模拟股票价格的波动、模拟随机游走等。下面是一个使用C语言中的rand()函数来生成符合均匀分布的随机数的代码示例:

#include <stdio.h>
#include <stdlib.h>
#include <time.h>int main() 
{srand((unsigned)time(NULL)); // 初始化随机数生成器for (int i = 0; i < 10; i++) {printf("%d ", rand()); // 生成随机数}printf("\n");return 0;
}

需要注意的是,在使用rand()函数时,需要先调用srand()函数初始化随机数种子,否则每次生成的随机数序列都是相同的。此外,由于rand()函数的随机数分布并不是完全均匀的,因此在某些模拟场景下需要使用其他的随机数生成器。

六、总结

在C语言中,生成随机数的方法主要有以下几种:

  1. 使用srand函数和rand函数结合生成伪随机数。srand函数可以设置随机数种子,rand函数则生成伪随机数。这种方法可以用于一些简单的随机需求。
  2. 使用time函数和rand函数结合生成伪随机数。time函数返回当前的时间戳,可以作为随机种子。这种方法可以用于需要更高随机性的场景。
  3. 使用rand_r函数生成可重入的伪随机数。这种方法可以在多线程环境下使用。
  4. 使用/dev/random或/dev/urandom设备文件生成真随机数。这种方法可以生成更安全的随机数。

本文主要介绍了前两种方法。

在实际应用中,生成随机数的场景非常多,比如:

  1. 生成验证码、随机密码等需要随机性的字符串。
  2. 模拟随机事件或者随机生成数据用于测试。
  3. 生成密钥、加密盐等需要高强度随机性的数据。
  4. 给定范围内随机生成数字,比如在游戏中生成随机数量的物品。

生成高质量的随机数对于许多计算机应用程序非常重要。随机数在密码学、模拟、游戏、金融交易、统计学等领域都有广泛的应用。

如果生成的随机数质量不好,可能会导致不可预测的结果和意外的行为。例如,密码学中使用的伪随机数生成器如果质量不好,可能会导致密码易受攻击,从而暴露敏感数据;在模拟中,不良的随机数可能导致结果不准确,无法反映真实情况。

因此,为了确保应用程序的安全性和正确性,生成高质量的随机数非常重要。这可以通过使用高质量的随机数生成器和正确地使用它们来实现。

相关文章:

掷骰子式的乐趣:探究C语言生成随机数的奥秘

掷骰子式的乐趣&#xff1a;探究C语言生成随机数的奥秘一、引言二、C标准库的rand函数三、srand函数的使用四、基于时间的种子生成五、高质量随机数的应用一、引言 C语言中生成随机数是一项非常重要的功能&#xff0c;因为许多现代应用程序需要使用随机数。随机数可以用于密码…...

一线大厂软件测试常见面试题1500问,背完直接拿捏面试官,

三、测试理论 3.1 你们原来项目的测试流程是怎么样的? 我们的测试流程主要有三个阶段&#xff1a;需求了解分析、测试准备、测试执行。 1、需求了解分析阶段 我们的SE会把需求文档给我们自己先去了解一到两天这样&#xff0c;之后我们会有一个需求澄清会议&#xff0c; 我…...

小迪安全day12WEB漏洞-SQL注入之简要SQL注入

小迪安全day12WEB漏洞-SQL注入之简要SQL注入 注入产生原理详细分析 可控变量带入数据库查询变量未存在过滤或过滤不严谨 连接符区分 and是sql语句连接符&#xff0c;&是uel参数连接符 and 11是注入语句&#xff0c; &是添加一个新变量 数据库内容 数据库A 网站…...

自动化测试学习(七)-正则表达式,你真的会用吗?

目录 一、正则表达式在python中如何使用 二、用正则表达式匹配更多模式 三、常用字符分类的缩写代码 总结 所谓正则表达式&#xff08;regex&#xff09;&#xff0c;就是一种模式匹配&#xff0c;学会用正则匹配&#xff0c;就可以达到事半功倍的效果。 一、正则表达式在…...

验证码——vue中后端返回的图片流如何显示

目录 前言 一、p调用接口获取验证码 canvas画布渲染&#xff1f; 二、后端返回图片&#xff08;图片流&#xff09;&#xff0c;前端显示 1.blob 2.arraybuffer 总结 前言 登录界面经常会有验证码&#xff0c;验证码的实现方式也有很多&#xff0c;我目前做过以下两种&…...

聚观早报 | 拼多多驳斥Google的指控;80%美国人工作将被AI影响

今日要闻&#xff1a;拼多多驳斥Google“恶意软件”的指控&#xff1b;80%美国人工作将被AI影响&#xff1b;iPhone 15 Pro设计图上热搜&#xff1b;贾扬清离职阿里投身AI大模型创业&#xff1b;OPPO Find X6 系列发布拼多多驳斥Google“恶意软件”的指控 3 月 21 日&#xff0…...

define,typedef,inline 的区别

define 1.用于在代码中创建宏定义&#xff0c;将一个标识符替换为一个表达式或语句。例如&#xff1a; #define PI 3.14159 #define SQUARE(x) ((x) * (x))这样&#xff0c;程序中所有出现的 PI 都将被替换为 3.14159&#xff0c;SQUARE(x) 则被替换成了 (x) * (x)。 使用 #…...

百度文心一言正式亮相

OpenAI 刚发布了 GPT-4&#xff0c;百度预热已久的人工智能生成式对话产品也终于亮相了。昨天下午&#xff0c;文心一言 (ERNIE Bot)—— 百度全新一代知识增强大语言模型、文心大模型家族的新成员&#xff0c;正式在百度总部 “挥手点江山” 会议室里发布。 发布会一开场&…...

使用Android架构模板

使用Android架构模板 项目介绍 为了方便开发者引入最新的Android架构组建进行开发&#xff0c;Google官方给我们引入了一个架构模板&#xff0c;方便我们快速进入开发。 github地址&#xff1a; https://github.com/android/architecture-templates 该模板遵循官方架构指南 …...

2023年天津市逆向re2.exe解析-比较难(超详细)

2023年天津市逆向re2.exe解析(较难) 1.拖进IDA里进行分析2.动态调试3.编写EXP脚本获取FLAG4.获得FLAG1.拖进IDA里进行分析 进入主程序查看伪代码 发现一个循环,根据行为初步判定为遍历输入的字符并对其ascii^7进行加密 初步判断sub_1400ab4ec为比较输入和flag的函数 跟进u…...

springboot: mybatis动态拼接sql查询条件

目录 需求01: 根据不同类型 查询不同的订单名, 1. 书写订单 类型转换方法 2. 使用方式: 3.. 构建条件构造器并进行查询, 传递查询参数 4. Mapper 写法 5. 最核心位置 xml位置 6. sql执行效果: 需求01: 根据不同类型 查询不同的订单名, 条件也是不同的, 需要复用sql…...

最优化算法 - 动态规划算法

动态规划算法简介 动态规划&#xff08;Dynamic programming&#xff09;是一种在数学、管理科学、计算机科学、经济学和生物信息学中使用的&#xff0c;通过把原问题分解为相对简单的子问题的方式求解复杂问题的方法。 动态规划常常适用于有重叠子问题和最优子结构性质的问题…...

springCloud学习【3】之Docker(1)

文章目录一 Docker环境准备1.1 应用部署的环境问题1.2 Docker简介1.3 Docker解决操作系统环境差异1.4 Docker和虚拟机的区别1.5 Docker架构1.5.1 镜像和容器1.5.2 DockerHub1.5.3 Docker架构1.5.4 Docker工作流1.6 Docker的安装和启动1.7 安装步骤1.8 启动Docker1.9 配置镜像加…...

难以置信,已经有人用 ChatGPT 做 Excel 报表了?

要问2023年初科技领域什么最火&#xff0c;那自然是 ChatGPT。 ChatGPT 由人工智能研究实验室 OpenAI 于2022年11月30日推出。上线短短5天&#xff0c;用户数量已突破100万&#xff0c;在今年2月份&#xff0c;用户数量已经突破1亿。 ChatGPT 是一个超级智能聊天机器人&#…...

中断相关操作函数HAL_NVIC_SetPriority()、HAL_NVIC_EnableIRQ()

文章目录HAL_NVIC_SetPriority()&#xff1a;设置中断优先级HAL_NVIC_EnableIRQ()&#xff1a;使能中断结束HAL_NVIC_SetPriority()&#xff1a;设置中断优先级 HAL_NVIC_SetPriority()函数是一个用于设置中断优先级的函数&#xff0c;其定义如下&#xff1a; void HAL_NVIC_…...

企业增长秘诀丨设立优质的帮助中心,加深用户产品使用深度,促进产品转化

客户的留存问题一直备受企业关注&#xff0c;留存率的高低反应了产品的真实状况&#xff0c;将直接影响企业后期的发展规划。下文将为大家剖析下产品中客户的转化流程&#xff0c;以及如何提高产品的使用深处与复购率。 产品中&#xff0c;从客户生命周期角度&#xff0c;可分…...

3.OSPF与BGP的联动

14.3实验3&#xff1a;OSPF与BGP联动配置 实验目的实验拓扑实验步骤 配置IP地址 AR1的配置 <Huawei>system-view Enter system view, return user view with CtrlZ. [Huawei]undo info-center enable Info: Information center is disabled. [Huawei]sysname AR1 …...

机器学习算法——决策树详解

文章目录前言&#xff1a;决策树的定义熵和信息熵的相关概念信息熵的简单理解经典的决策树算法ID3算法划分选择或划分标准——信息增益ID3算法的优缺点C4.5算法信息增益率划分选择或划分标准——Gini系数&#xff08;CART算法&#xff09;Gini系数计算举例CART算法的优缺点其他…...

配置Jenkins

目录 一.前言 1.1简介 1.2工作步骤图 二.配置jenkins部署项目 2.1项目新建 2.2jenkinsfile文件如下 三.jenkins工作台配置 3.1.点击新建item进入新建页面,输入任务名称,选择pipeline 3.2.选择第二个配置 3.4将ideal中jenkinsfile文件的路径粘入脚本路径中 3.5启动项目…...

【STL三】序列容器——array容器

【STL三】序列容器——array一、array简介二、头文件三、模板类四、成员函数1、迭代器2、元素访问3、容量4、操作五、demo1、容量&#xff08;不使用迭代器&#xff09;2、使用迭代器3、元素访问 at()、front()、back()、data()一、array简介 array 容器是 C 11 标准中新增的序…...

【STL四】序列容器——vector容器

【STL容器】序列容器——vector容器一、简介二、头文件三、模板类四、成员函数1、迭代器2、元素访问3、容量4、修改操作五、demo1、容量reserve、capacity、shrink_to_fit2、修改操作pop_back()、push_back3、修改操作insert()4、修改操作emplace()5、修改操作erase()、swap()、…...

4年功能测试,我一进阶python接口自动化测试,跳槽拿了20k......

目录&#xff1a;导读前言一、Python编程入门到精通二、接口自动化项目实战三、Web自动化项目实战四、App自动化项目实战五、一线大厂简历六、测试开发DevOps体系七、常用自动化测试工具八、JMeter性能测试九、总结&#xff08;尾部小惊喜&#xff09;前言 很多人在这求职市场…...

基于Pytorch的可视化工具

深度学习网络通常具有很深的层次结构&#xff0c;而且层与层之间通常会有并联、串联等连接方式。当使用PyTorch建立一个深度学习网络并输出文本向读者展示网络的连接方式是非常低效的&#xff0c;所以需要有效的工具将建立的深度学习网络结构有层次化的展示&#xff0c;这就需要…...

XCPC第十一站,带你学会图论基本算法

我们约定&#xff1a;以下n表示点的数目&#xff0c;m表示边的数目。 引子1——邻接表存储图的方法&#xff08;&#xff09;&#xff08;暂时不考虑重边和自环&#xff09; 现在我们有n个点&#xff08;编号为1~n&#xff09;和m条边&#xff0c;要用数组存储它们&#xff0c…...

【MySQL】1 MySQL的下载、安装与配置|提供安装包

欢迎来到爱书不爱输的程序猿的博客, 本博客致力于知识分享&#xff0c;与更多的人进行学习交流 目前,已开了以下专栏,欢迎关注与指导 1️⃣Java基础知识系统学习(持续更文中…) 2️⃣UML(已更完) 3️⃣MySQL(持续更文中…) MYSQL的下载、安装与配置1.下载MySQL5.71.1安装包的获…...

Redis 事务

目录Redis 事务一、Redis事务的概念&#xff1a;二、redis事务提出的逻辑&#xff1a;三、redis事务的基本操作四、事务的执行流程五、redis锁六、redis分布式锁Redis 事务 一、Redis事务的概念&#xff1a; Redis 事务的本质是一组命令的集合。事务支持一次执行多个命令&…...

【linux】:进程控制

文章目录 前言一、什么是写时拷贝二、进程控制 1.进程终止2.进程等待三丶进程程序替换总结前言 了解上一篇文章中的进程地址空间后&#xff0c;我们再来说说进程控制的概念&#xff0c;进程控制我们需要搞清楚三个问题&#xff1a;如何进程终止&#xff0c;如何解决僵尸进程问…...

在recyclerview中使用其item布局的ViewBinding类需要注意的问题

问题描述 最近在使用RecycerView的瀑布流布局&#xff0c;我想直接用ViewBinding取得item中的一个TextView然后根据position进行赋值。 比如我点击测试标题2&#xff0c;它在日志中应该能打印出测试标题2才对。 但是他却打印出“测试标题0” 按理来说标题应该更点击的位置对…...

基于EB工具的TC3xx_MCAL配置开发05_ADC模块硬件Pwm触发Demo配置

目录 1.概述2. Pwm相关配置2.1 PWM->General配置2.2 PWM->PwmChannel配置2.2.1 Reference Pwm通道配置2.2.2 触发ADC采集的Pwm通道配置3. ADC相关配置3.1 AdcHwExtTrigSelect配置4. MCU相关配置4.1 GTM配置4.1.1 McuGtmTomAllocationConf配置4.1.2 GtmTriggerForAdc配置1…...

Qt示例3:用Qt画一个温度计

示例1 以下是用Qt绘制一个简单的温度计的示例代码&#xff1a; #include <QPainter> #include <QWidget> #include <QApplication> class Thermometer : public QWidget { public:Thermometer(QWidget *parent 0); protected:void paintEvent(QPaintEvent …...