STL系列十一 随机三趣题——随机重排,文件中随机取一行,生成N个随机数

2024-04-23 01:58

本文主要是介绍STL系列十一 随机三趣题——随机重排,文件中随机取一行,生成N个随机数,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

8人阅读 评论(0) 收藏 举报

目录(?)[+]

本文将介绍三个有趣的随机问题,分别是随机重新排列、从文件中随机取一行数据、生成N个随机数。

 

一.随机重新排列

将一个序列打乱并对其进行随机的重新排列,关键在于每种序列的被选择概率要一样,不然有失“公平”。现在让我们来寻找如何保证每种序列被选择的概率一样大的算法。

首先假设这个数组只有二个元素,设数组a{1, 2},显然这个数组只有二种可能的排列,要么是{12}要么是{21}。很容易想到一种方法——只要第二个元素有50%的概率与第一个元素交换即可。用代码表现下:

if (rand() % 2 == 0)

      swap(a[0], a[1])

由于rand() % 2的结果要么为0,要么为1,且各占50%的概率。因此swap(a[0],a[1])的执行概率也是50%,如果执行了,结果会是{21}。没有执行,结果会是{12}。所以这样两种排列出现的可能均为50%

 

接下来再假充这个数组有三个元素设数组a{1, 23},这个数组有六种可能的排列,要么是{123}{132}{213}{231}{312}{321}。这么多的排列看起来好象有点复杂,不知道从何下手。其实结合上面的分析,我们可以这样考虑:

1.先调整前二个元素即{1, 23}先生成{123}{213}

2.然后对{123},第三个元素以1/3的概率与第一,第二,第三个元素进行交换就可以等概率的得到{321}{132}{123}

3.同理对{213},可以等概率的得到{312}{231}{213}

根据贝叶斯公式,不难计算出由这些排列出现的可能性都是1/2 * 1/3 = 1/6。完全符合每种序列的被选择概率相同的要求。

 

这样随机重新排列的算法就找到了,下面给出示意代码:

[cpp] view plain copy
  1. //随机重新排列  
  2. //参照STL中的random_shuffle  
  3. #include <stdio.h>  
  4. #include <stdlib.h>  
  5. #include <time.h>  
  6. inline void Swap(int *a, int *b)  
  7. {  
  8.     int c = *a;  
  9.     *a = *b;  
  10.     *b = c;  
  11. }  
  12. //随机重新排列函数  
  13. void Random_shuffle(int a[], int n)  
  14. {  
  15.     srand(time(NULL));  
  16.     for (int i = 1; i < n; i++)  
  17.         Swap(&a[i], &a[rand() % (i + 1)]);  
  18. }  
  19. void PrintfIntArray(int a[], int n)  
  20. {  
  21.     for (int i = 0; i < n; i++)  
  22.         printf("%d ", a[i]);  
  23.     putchar('\n');  
  24. }  
  25. int main()  
  26. {  
  27.     printf("           随机重新排列 \n");  
  28.     printf("--- by MoreWindows( http://blog.csdn.net/MoreWindows )  ---\n\n");  
  29.   
  30.     const int MAXN = 8;  
  31.     int a[MAXN] = {1, 2, 3, 4, 5, 6, 7, 8};  
  32.   
  33.     printf("原数组:\n");  
  34.     PrintfIntArray(a, MAXN);  
  35.   
  36.     Random_shuffle(a, MAXN);  
  37.   
  38.     printf("随机重新排列后:\n");  
  39.     PrintfIntArray(a, MAXN);  
  40.     return 0;  
  41. }  

运行结果如下:

有兴趣的童鞋可以用STL中的random_shuffle()来改写上面的程序,相信知道其实现原理后会对random_shuffle()有更深的认识。

另外,这种按顺序先后来交换元素得到新排列的方法与生成全排列非常类似,可以参考《STL系列之十 全排列(百度迅雷笔试题)》然后对比下两种方法在思路上相似之处。

 

二.从文件中随机取一行数据

如果先统计文件有多少行,再根据rand() % 行数选择对应行也是可以行的,但效率显然会有点低了。有没有一种方法可以只遍历文件一次了?请看代码:

[cpp] view plain copy
  1. //从文件中取机选取一行  
  2. #include <stdio.h>  
  3. #include <stdlib.h>  
  4. #include <time.h>  
  5. int main()  
  6. {  
  7.     printf("           从文件中取机选取一行 \n");  
  8.     printf("--- by MoreWindows( http://blog.csdn.net/MoreWindows )  ---\n\n");  
  9.   
  10.     int i, num, nChooseNum;  
  11.     const char strFileName[] = "in.txt";  
  12.       
  13.     freopen(strFileName, "r", stdin);  
  14.     srand(time(NULL));  
  15.     i = 1;  
  16.     while (scanf("%d", &num) != EOF)  
  17.     {  
  18.         if (rand() % i == 0)  
  19.             nChooseNum = num;  
  20.         i++;  
  21.     }  
  22.     printf("从文件中选取出: %d\n", nChooseNum);  
  23.     return 0;  
  24. }  

运行结果如下(in.txt中为09,每个数字占一行):

对代码进行下讲解,以三行数据为例,首先对文本的第一行,rand() % 1,结果必然为0。所以第一行已被选中了,然后对第二行,rand()%2,结果要么为0,要么为1。故第二行有 50的可能性被选中,然后对第三行,rand()%3,显然被选中的概率为1/3。故有:

选中第一行的概率为1 * 1/2 * 2/3 = 1/3

选中第二行的概率为1/2 * 2/3 = 1/3

选中第三行的概率为1/3

故每一行被选中是等概率的。

 

三.生成N个随机数

这个有很多方法,常见方法用set记录已经生成的数据,然后判断set的大小是否符合要求。代码如下所示:

[cpp] view plain copy
  1. //生成n个指定范围的随机数  
  2. #include <stdio.h>  
  3. #include <time.h>  
  4. #include <set>  
  5. //生成n个[s, e)范围的数  
  6. void GetRandNumberInRange(int *a, int n, int s, int e)  
  7. {  
  8.     int   i, j;  
  9.     std::set<int>  m;  
  10.     std::set<int>::iterator setpos;  
  11.       
  12.     srand(time(NULL));    
  13.     while (m.size() < n)  
  14.     {  
  15.         j = rand() % (e - s) + s;  
  16.         m.insert(j);  
  17.     }  
  18.       
  19.     i = 0;  
  20.     for (setpos = m.begin(); setpos != m.end(); setpos++)  
  21.         a[i++] = *setpos;  
  22. }  
  23. void PrintfIntArray(int a[], int n)  
  24. {  
  25.     for (int i = 0; i < n; i++)  
  26.         printf("%d ", a[i]);  
  27.     putchar('\n');  
  28. }  
  29. int main()  
  30. {  
  31.     const int NMAX = 20;  
  32.     const int NUMSTART = 1;  
  33.     const int NUMEND = 100;  
  34.   
  35.     printf("           生成%d个%d到%d之间的数 \n", NMAX, NUMSTART, NUMEND);  
  36.     printf("--- by MoreWindows( http://blog.csdn.net/MoreWindows )  ---\n\n");    
  37.   
  38.     int    a[NMAX];  
  39.       
  40.     GetRandNumberInRange(a, NMAX, NUMSTART, NUMEND + 1);  
  41.     PrintfIntArray(a, NMAX);  
  42.     return 0;  
  43. }  

运行结果如下:

这种方法会导致调用随机函数次数过多,从而效率低下。可以采用一种改进的方法,如要生成3110之间的数,取10-3=7,因此可以先t=rand()%7,保存t,然后再t=rand()%8,如果t这个数已经存在就取8保存,再t=rand()%9,同样如果t这个数已经存在就取9保存。这样只会调用3次随机函数。当然这种方法不能完全保证每个数字被选择概率相同,算是牺牲“公平”来保证效率吧。

生成N个随机数的改进方法代码如下:

[cpp] view plain copy
  1. //生成n个指定范围的随机数  
  2. #include <set>  
  3. #include <cstdio>  
  4. #include <cstdlib>  
  5. #include <ctime>  
  6. //在[s, e)区间上随机取n个数并存放到a[]中  
  7. void GetRandomNum(int *a, int n, int s, int e)  
  8. {  
  9.     std::set<int> set_a;  
  10.     srand(time(NULL));  
  11.     for (int i = e - n; i < e; i++)  
  12.     {  
  13.         int num = (rand() % (i - s)) + s;  
  14.         if (set_a.find(num) == set_a.end())  
  15.             set_a.insert(num);  
  16.         else  
  17.             set_a.insert(i);  
  18.     }  
  19.     i = 0;  
  20.     std::set<int>::iterator pos;  
  21.     for (pos = set_a.begin(); pos != set_a.end(); pos++)  
  22.         a[i++] = *pos;  
  23. }  
  24. void PrintfIntArray(int a[], int n)  
  25. {  
  26.     for (int i = 0; i < n; i++)  
  27.         printf("%d ", a[i]);  
  28.     putchar('\n');  
  29. }  
  30. int main()  
  31. {  
  32.     const int NMAX = 20;  
  33.     const int NUMSTART = 1;  
  34.     const int NUMEND = 100;  
  35.       
  36.     printf("           生成%d个%d到%d之间的数 改进版\n", NMAX, NUMSTART, NUMEND);  
  37.     printf("--- by MoreWindows( http://blog.csdn.net/MoreWindows )  ---\n\n");   
  38.     int a[NMAX];  
  39.       
  40.     GetRandomNum(a, NMAX, NUMSTART, NUMEND + 1);  
  41.     PrintfIntArray(a, NMAX);  
  42.     return 0;  
  43. }  

运行结果如下:

 

有关随机的趣味题就介绍到这里,有任何问题欢迎和我交流,直接留言或发送邮件均可。邮箱:morewindows@126.com

 

下一篇《STL系列十二链表排序的6次改进(对《STL源码剖析》作个补充)》将会对STL中的链表排序进行详细讲解,如果看过《STL源码剖析》书中142页的链表排序代码将会更有收获。更多内容,请继续关注MoreWindows

这篇关于STL系列十一 随机三趣题——随机重排,文件中随机取一行,生成N个随机数的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/927473

相关文章

浅析如何使用Swagger生成带权限控制的API文档

《浅析如何使用Swagger生成带权限控制的API文档》当涉及到权限控制时,如何生成既安全又详细的API文档就成了一个关键问题,所以这篇文章小编就来和大家好好聊聊如何用Swagger来生成带有... 目录准备工作配置 Swagger权限控制给 API 加上权限注解查看文档注意事项在咱们的开发工作里,API

Java使用POI-TL和JFreeChart动态生成Word报告

《Java使用POI-TL和JFreeChart动态生成Word报告》本文介绍了使用POI-TL和JFreeChart生成包含动态数据和图表的Word报告的方法,并分享了实际开发中的踩坑经验,通过代码... 目录前言一、需求背景二、方案分析三、 POI-TL + JFreeChart 实现3.1 Maven

MybatisGenerator文件生成不出对应文件的问题

《MybatisGenerator文件生成不出对应文件的问题》本文介绍了使用MybatisGenerator生成文件时遇到的问题及解决方法,主要步骤包括检查目标表是否存在、是否能连接到数据库、配置生成... 目录MyBATisGenerator 文件生成不出对应文件先在项目结构里引入“targetProje

Python使用qrcode库实现生成二维码的操作指南

《Python使用qrcode库实现生成二维码的操作指南》二维码是一种广泛使用的二维条码,因其高效的数据存储能力和易于扫描的特点,广泛应用于支付、身份验证、营销推广等领域,Pythonqrcode库是... 目录一、安装 python qrcode 库二、基本使用方法1. 生成简单二维码2. 生成带 Log

Python中的随机森林算法与实战

《Python中的随机森林算法与实战》本文详细介绍了随机森林算法,包括其原理、实现步骤、分类和回归案例,并讨论了其优点和缺点,通过面向对象编程实现了一个简单的随机森林模型,并应用于鸢尾花分类和波士顿房... 目录1、随机森林算法概述2、随机森林的原理3、实现步骤4、分类案例:使用随机森林预测鸢尾花品种4.1

Python使用Pandas库将Excel数据叠加生成新DataFrame的操作指南

《Python使用Pandas库将Excel数据叠加生成新DataFrame的操作指南》在日常数据处理工作中,我们经常需要将不同Excel文档中的数据整合到一个新的DataFrame中,以便进行进一步... 目录一、准备工作二、读取Excel文件三、数据叠加四、处理重复数据(可选)五、保存新DataFram

SpringBoot生成和操作PDF的代码详解

《SpringBoot生成和操作PDF的代码详解》本文主要介绍了在SpringBoot项目下,通过代码和操作步骤,详细的介绍了如何操作PDF,希望可以帮助到准备通过JAVA操作PDF的你,项目框架用的... 目录本文简介PDF文件简介代码实现PDF操作基于PDF模板生成,并下载完全基于代码生成,并保存合并P

使用C#如何创建人名或其他物体随机分组

《使用C#如何创建人名或其他物体随机分组》文章描述了一个随机分配人员到多个团队的代码示例,包括将人员列表随机化并根据组数分配到不同组,最后按组号排序显示结果... 目录C#创建人名或其他物体随机分组此示例使用以下代码将人员分配到组代码首先将lstPeople ListBox总结C#创建人名或其他物体随机分组

详解Java中如何使用JFreeChart生成甘特图

《详解Java中如何使用JFreeChart生成甘特图》甘特图是一种流行的项目管理工具,用于显示项目的进度和任务分配,在Java开发中,JFreeChart是一个强大的开源图表库,能够生成各种类型的图... 目录引言一、JFreeChart简介二、准备工作三、创建甘特图1. 定义数据集2. 创建甘特图3.

Spring Security 从入门到进阶系列教程

Spring Security 入门系列 《保护 Web 应用的安全》 《Spring-Security-入门(一):登录与退出》 《Spring-Security-入门(二):基于数据库验证》 《Spring-Security-入门(三):密码加密》 《Spring-Security-入门(四):自定义-Filter》 《Spring-Security-入门(五):在 Sprin