新旧C++生成随机浮点数方法,你喜欢哪个?

点击蓝字

805f4e868778602fdffe7df5a53bf545.png

关注我们

一、在C++11之前,我们通常采用rand函数来生成随机数。

有时我们想用rand生成一组随机数,即使我们调用了srand,但生成的还是相同值。为什么会产生这种情况?又该如何解决?下面将用第一视角一起探究这其中的奥秘。

场景描述:

想生成一组整形随机数,放入数组中,用来测试自己的排序是否正确。

于是我写出了下方代码,生成随机数。

先简单了解下用到的函数:

//返回time_t类型的 当前时间的时间戳
time_t time (time_t* timer);//传入一个种子,为伪随机数生成器初始化
void srand (unsigned int seed);//得到一个整形伪随机数
int rand (void);
#include <stdio.h>
#include <time.h>
#include <stdlib.h>int main()
{int arr[10] = { 0 };for (int i = 0; i < 10; ++i){srand((unsigned int)time(NULL));//两个相减是为了出现负的随机数,使测试范围更广arr[i] = (rand() % 100 + 1) - (rand() % 100 + 1);printf("%d ", arr[i]);}return 0;
}

我发现尽管我调用了srand函数,可生成的数组值还是同一个。我思考后想到,因为for循环执行速度太快,整个程序都是在一秒内完成的。所以出现了都是同一个值的情况。

初步解决

于是我想出了下面的解决方法:

我可以在for循环内调用Sleep函数,让我的电脑休眠一下,这样就不会出现上述情况了。

于是我写出了下方的代码:

#include <stdio.h>
#include <time.h>
#include <stdlib.h>
#include <windows.h>int main()
{int arr[10] = { 0 };for (int i = 0; i < 10; ++i){Sleep(1000);srand((unsigned int)time(NULL));arr[i] = (rand() % 100 + 1) - (rand() % 100 + 1);printf("%d ", arr[i]);}return 0;
}

通过休眠后,就成功解决问题了。可是,

如果睡眠时间太短,那么还是会出现重复的现象;

如果睡眠时间太长,程序运行速度就太慢。

最终方法

因为上述的原因,又详细了解了一下rand和srand的基本原理,最终成功解决了该问题。

给srand函数传入一个数值后,srand会根据这个生成一个随机序列表(通常有4,294,967,296个数),传入相同的数生成的序列表是相同的。然后rand从序列的头部取出一个数返回,然后将这个数放在随机序列表尾部,因此如果你要取的数据量非常大,是会出现与之前取出的数重复的情况。

此时,上面出现的问题也很好解决了。因为计算机运行速度很快,所以我们每次进入循环都会生成一个相同的随机序列表,rand函数只会取出其第一个数。

要解决这个问题,我们只需要在循环前调用一次srand函数就好了,这样就不会重复生成序列表了。

下方是最终形式的代码:

#include <stdio.h>
#include <time.h>
#include <stdlib.h>int main()
{int arr[10] = { 0 };srand((unsigned int)time(NULL));for (int i = 0; i < 10; ++i){arr[i] = (rand() % 100 + 1) - (rand() % 100 + 1);printf("%d ", arr[i]);}return 0;
}

但rand函数对一些情况显得难以处理:

  • 不同范围的随机数

  • 需要随机浮点数

  • 需要非均匀分布的随机数

二、C++11定义在头文件random中的随机数库

下文将使用C++11定义在头文件random中的随机数库通过一组协作的类来解决这些问题:随机数引擎类和随机数分布类。

  • 一个引擎类可以生成unsigned随机数序列

  • 一个分布类使用一个引擎类生成指定类型的、在给定范围内的、服从特定概率分布的随机数

1、生成等概率随机数

生成随机整数

uniform_int_distribution:产生均匀分布的整数

template <class IntType = int> 
class uniform_int_distribution;// IntType
// An integer type. Aliased as member type result_type.
// By default, this is int.
#include <iostream>
#include <random>
#include <ctime>
using namespace std;int main()
{//产生[1, 100]左闭右闭区间的随机整数uniform_int_distribution<int> u(1, 100);default_random_engine e;//为随机数引擎设置随机种子,若不设置每次生成的随机数相同(可以创建时设置)//类似srand的用法,相同的种子生成的随机数相同//default_random_engine e(time(NULL));e.seed(time(NULL));for (size_t i = 0; i < 10; ++i){cout << u(e) << " ";}cout << endl;return 0;
}

生成随机浮点数

uniform_real_distribution:产生均匀分布的实数

template <class RealType = double> 
class uniform_real_distribution;// RealType
// A floating-point type. Aliased as member type result_type.
// By default, this is double.
#include <iostream>
#include <random>
#include <ctime>
using namespace std;int main()
{//生成[-1, 1]范围随机浮点数//模板参数只能是浮点类型(float,double, long double)uniform_real_distribution<double> u(-1, 1);default_random_engine e(time(NULL));for (size_t i = 0; i < 10; ++i){cout << u(e) << " ";}cout << endl;return 0;
}

2、生成非均匀分布随机数


正态分布随机数

template <class RealType = double> 
class normal_distribution;
#include <iostream>
#include <random>
#include <ctime>
using namespace std;int main()
{
//生成符合均值为10,标准差为2的随机数normal_distribution<double> u(10, 2);
default_random_engine e(time(NULL));for (size_t i = 1; i <= 100; ++i){
printf("%-9.6lf ", u(e));
if (i % 10 == 0){
cout << endl;}}
cout << endl;return 0;
}

二项分布的布尔值

class bernoulli_distribution;
#include <iostream>
#include <random>
#include <ctime>
using namespace std;int main()
{
// 生成1的概率为0.7
bernoulli_distribution u(0.7); 
default_random_engine e(time(NULL));for (int i = 0; i < 10; i++) {
cout << u(e) << " ";}
cout << endl;return 0;
}

简单分享,快乐学习,如有错误请多包涵!

*声明:本文于网络整理,版权归原作者所有,如来源信息有误或侵犯权益,请联系我们删除或授权事宜。

941bdacdd565ec106c86a5a905be61d3.png

6a7a9d8c665a8f6cb78681111ef2fd03.gif

戳“阅读原文”我们一起进步

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/337608.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

html页面加载完成后会触发的事件_前端隐秘角落 - 页面渲染

前言如图所示&#xff0c;webkit内核浏览器的渲染过程(解析HTML&#xff0c;构建DOM树&#xff0c;解析CSS&#xff0c;构建CSSOM树 &#xff0c;构建render树&#xff0c;布局layout&#xff0c;绘制painting)&#xff0c;这些过程理解起来可能有些抽象&#xff0c;今天我们一…

计算机进管理提示找不到入口,win10系统开机提示xxxdll模块已加载但找不到入口点的教程...

有关win10系统开机提示xxxdll模块已加载但找不到入口点的操作方法想必大家有所耳闻。但是能够对win10系统开机提示xxxdll模块已加载但找不到入口点进行实际操作的人却不多。其实解决win10系统开机提示xxxdll模块已加载但找不到入口点的问题也不是难事&#xff0c;小编这里提示两…

十大经典排序,你真的都会了吗?(源码详解)

点击蓝字关注我们一、前言&#xff1a;排序的概念排序&#xff1a;所谓排序&#xff0c;就是使一串记录&#xff0c;按照其中的某个或某些关键字的大小&#xff0c;递增或递减的排列起来的操作。稳定性&#xff1a;假定在待排序的记录序列中&#xff0c;存在多个具有相同的关键…

jvm 架构_不可变的基础架构,热部署和JVM

jvm 架构您是否在生产中部署和取消部署基于JVM的应用程序&#xff08;无论JVM容器/无容器&#xff09;&#xff1f; 也就是说&#xff0c;当您拥有某个应用程序或服务的新版本时&#xff0c;是否通过“取消部署”和“热部署”该应用程序的新更新版本来更改正在运行的JVM&#x…

c语言默认参数_5.1 C++有默认参数的函数

点击上方“C语言入门到精通”&#xff0c;选择置顶第一时间关注程序猿身边的故事作者闫小林白天搬砖&#xff0c;晚上做梦。我有故事&#xff0c;你有酒么&#xff1f;C有默认参数的函数在函数调用时形参从实参获取值&#xff0c;因为实参的个数要和形参相同&#xff0c;但有时…

计算机组成原理唐朔飞课后答案第六章,计算机组成原理第六章部分课后题答案(唐朔飞版)...

计算机组成原理第六章部分课后题答案(唐朔飞版) 6.4 设机器数字‎长为8位(含1位符号‎位在内)&#xff0c;写出对应下‎列各真值的‎原码、补码和反码‎。 -13/64&#xff0c;29/128&#xff0c;100&#xff0c;-87 解&#xff1a;十进制数 二进制数 原 码 反 码 补 码 -13/64 …

jboss eap 7.0_是时候抛弃Java 7 – JBoss EAP 6.4了!

jboss eap 7.0这一周真是太棒了。 JBoss EAP 6.4已发布&#xff0c;在众多技术增强和新功能中 &#xff0c;最大的是&#xff1a;Java 8已添加到受支持的配置列表中。 其中包括Oracle JDK和IBM JDK。 Java SE 7公开更新结束通知 2015年4月之后&#xff0c;Oracle将不再将Java …

C语言史上最愚蠢的BUG ???

点击蓝字关注我们本文来自“The most stupid C bug ever”&#xff0c;很有意思&#xff0c;分享给大家。我相信这样的bug&#xff0c;就算你是高手你也会犯的。你来看看作者犯的这个Bug吧。。首先&#xff0c;作者想用一段程序来创建一个文件&#xff0c;如果有文件名的话&…

python 字符串转日期_我总结的130页Python与机器学习之路V1.2.pdf,都是干货!

告别枯燥&#xff0c;通过学习有趣的小例子&#xff0c;扎实而系统的入门Python&#xff0c;从菜鸟到大师&#xff0c;个人觉得这是很靠谱的一种方法。通过一个又一个的小例子&#xff0c;真正领悟Python之强大&#xff0c;之简洁&#xff0c;真正做到高效使用Python.两周前发出…

大学计算机需要论文吗,大一新生刚开学,是否有必要带电脑?听听辅导员的建议,非常中肯...

原标题&#xff1a;大一新生刚开学&#xff0c;是否有必要带电脑&#xff1f;听听辅导员的建议&#xff0c;非常中肯各大高校的录取工作正在如火如荼的进行&#xff0c;很快考生们就能接到来自各个学校的录取通知书。对于考生来说&#xff0c;没有什么事情会比被心仪的大学录取…

计算机系统是连续系统,连续系统的计算机模拟

连续系统的计算机模拟 (36页)本资源提供全文预览&#xff0c;点击全文预览即可全文预览,如果喜欢文档就下载吧&#xff0c;查找使用更方便哦&#xff01;29.9 积分&#xfeff; 第2章 连续系统的计算机模拟本章讨论连续系统的模拟技术,由于这类系统中状态随时间连续动态地变化&…

ae合成设置快捷键_教程|AE教程第三波:必须掌握的关键帧之基础设置

该如何高效的学习AE&#xff1f;星驰君认为知其然还需知其所以然正确的学习顺序应该是了解核心原理&#xff0c;掌握基本操作&#xff0c;案例实战模仿比如&#xff0c;想要更好的掌握和运用关键帧来制作更复杂的效果。就先要知道关键帧是什么关键帧&#xff1a;计算机动画术语…

老兵精讲:高质量C语言编程的10条规范

点击蓝字关注我们C语言编码规范10条分享给大家&#xff0c;还是可以规避掉很多bug的&#xff01;1、最重要的规则编写代码时最重要的一条规则是&#xff1a;检查周围的代码并尝试模仿它。作为维护人员&#xff0c;如果收到的补丁明显与周围代码的编码风格不同&#xff0c;这是令…

springboot需要tomcat服务器吗_SpringBoot学习(二):内嵌服务器引擎(Tomcat,Jetty)实现原理...

概述SpringBoot使用main方法启动的一个重要特性是&#xff0c;不需要打包成war部署到Tomcat这种Servlet容器中&#xff0c;而是只需打包成jar&#xff0c;然后通过java或mvn等命令运行这个jar包&#xff0c;然后应用就可以在指定的端口监听客户端的连接请求了。在SpringBoot内部…

java8 函数式编程_使用Javaslang进行Java 8中的函数式编程

java8 函数式编程我们非常高兴地在jOOQ博客上宣布一个客座帖子&#xff0c;该帖子由HSH Nordbank的高级软件工程师Daniel Dietrich &#xff08;三人的丈夫和父亲&#xff09;撰写。 他目前作为项目负责人和首席开发人员为金融产品创建定价框架。 除工作外&#xff0c;他还对编…

while(1) 和 for(;;)有什么区别?

点击蓝字关注我们有粉丝问了类似这样的问题&#xff1a;while(1) 和 for(;;)它们不都是无限循环吗&#xff0c;作用应该一样啊&#xff0c;它们到底有什么区别&#xff1f;要回答这个问题&#xff0c;其实你各自编写一段while(1) 和 for(;;)的代码&#xff0c;编译对比一下代码…

C++高阶必会操作--模板元编程

点击蓝字关注我们泛型编程大家应该都很熟悉了&#xff0c;主要就是利用模板实现“安全的宏”&#xff0c;而模板元编程区别于我们所知道的泛型编程&#xff0c;它是一种较为复杂的模板&#xff0c;属于C的高阶操作了&#xff0c;它最主要的优点就在于把计算过程提前到编译期&am…

怎样做远程计算机控制系统,qq远程控制,怎样进行远程控制制作步骤

怎样设置qq远程控制&#xff1f;qq上有一个远程协助不少人都用过&#xff0c;但是这个远程小编建议不是认识的朋友&#xff0c;不要随便开放&#xff0c;很容易被窃取电脑资料&#xff0c;如果是不是很信得过有人&#xff0c;在申请协助时一定要自己盯着电脑&#xff0c;以防别…

pcl中ransac提取直线_复杂场景中的一个图像配准思路

在很多时候&#xff0c;我们可能需要使用到图像的识别与配准工作&#xff0c;来判断某个特征或者是划出某个特定特征的位置。现在的深度学习已经能够比较好地解决这个问题&#xff0c;比如常见的YOLO&#xff0c;可以利用几行设定代码就能够划出所需要识别的位置。但是精准度可…

程序员必知的10个C语言技巧

点击蓝字关注我们硬件设计师最常见的工作内容&#xff0c;就是通过写代码来测试硬件。这10个C语言技巧&#xff08;C语言仍然是常见的选择&#xff09;可以帮助设计师避免因基础性错误而导致某些缺陷的产生&#xff0c;并造成维护方面的困扰。为了成功的推出一个产品&#xff0…