两种经典方法解决随机数生成问题--利用rand5得到rand7

2023-12-20 14:36

本文主要是介绍两种经典方法解决随机数生成问题--利用rand5得到rand7,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

两种方法:

  • 等概率得到0和1,然后利用二进制移位运算生成随机数
  • 基于N * (randN - 1) + randN,可以等概率地生成1-N*N之间的数字

利用rand5得到rand7 为例说明

# 一个函数可以实现1-5的随机数,设计另一个函数,实现1-7的随机数
class Rand2rand:def f5(self):return int(random.random() * 5) + 1# 等概率得到0和1def g01(self):while True:tmp = self.f5()if tmp < 3:return 0elif tmp > 3:return 1else:continue# 等概率得到0-6def f06(self):while True:tmp = (self.g01() << 2) + (self.g01() << 1) + (self.g01() << 0)if tmp == 7:continueelse:return tmp# 等概率得到1-7def f17(self):return self.f06() + 1# 验证结果def testRes(self, testTimes: int, rand_generator):arr = [0] * 7for i in range(testTimes):tmp = rand_generator()arr[tmp - 1] += 1for i in range(len(arr)):print(f'{i + 1} 出现了 {arr[i]} 次')def f17_2(self):# 5*(self.f5()-1) 等概率产生 0, 5, 10, 15, 20# self.f5() 等概率产生 1,2,3,4,5# 想加后,等概率产生 1-25# rand5()可以得到rand25()# rand25()可以得到rand625()# 总结,若randN() 可以随机生成1-N,那么N*(randN()-1) + randN() 可以随机生成 1-N**2randVal = 22while randVal > 21:randVal = 5 * (self.f5() - 1) + self.f5()return 1 + randVal % 7# 结果验证
if __name__ == '__main__':obj = Rand2rand()rand_gen1 = obj.f17rand_gen2 = obj.f17_2obj.testRes(1000000, rand_gen2)

class Rand2randGeneral:"""更一般的问题,给定一个可以产出[min, max]之间的等概率随机数函数,生成一个可以等概率生成[from, to]之间的随机数函数"""def __init__(self, min, max):self.min = minself.max = maxdef randomBox(self):diff = int(random.random() * (self.max - self.min + 1))randomValue = self.min + diffreturn randomValuedef getrand01(self):size = self.max - self.min + 1# 判断size 是奇数还是偶数isOdd = True if (size & 1) != 0 else Falsemid = int(size / 2)while True:ans = self.randomBox() - self.minif isOdd and ans == mid:continueelif ans < mid:return 0else:return 1def getRandGeneral(self, fromValue, toValue):if fromValue == toValue:return fromValuerangeVal = toValue - fromValuenum = 1while (1 << num) - 1 < rangeVal:num += 1while True:ans = 0for i in range(num):ans |= self.getrand01() << iif ans > rangeVal:continueelse:return ans + fromValue

这篇关于两种经典方法解决随机数生成问题--利用rand5得到rand7的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

MySQL查询JSON数组字段包含特定字符串的方法

《MySQL查询JSON数组字段包含特定字符串的方法》在MySQL数据库中,当某个字段存储的是JSON数组,需要查询数组中包含特定字符串的记录时传统的LIKE语句无法直接使用,下面小编就为大家介绍两种... 目录问题背景解决方案对比1. 精确匹配方案(推荐)2. 模糊匹配方案参数化查询示例使用场景建议性能优

Java 线程安全与 volatile与单例模式问题及解决方案

《Java线程安全与volatile与单例模式问题及解决方案》文章主要讲解线程安全问题的五个成因(调度随机、变量修改、非原子操作、内存可见性、指令重排序)及解决方案,强调使用volatile关键字... 目录什么是线程安全线程安全问题的产生与解决方案线程的调度是随机的多个线程对同一个变量进行修改线程的修改操

关于集合与数组转换实现方法

《关于集合与数组转换实现方法》:本文主要介绍关于集合与数组转换实现方法,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1、Arrays.asList()1.1、方法作用1.2、内部实现1.3、修改元素的影响1.4、注意事项2、list.toArray()2.1、方

Python中注释使用方法举例详解

《Python中注释使用方法举例详解》在Python编程语言中注释是必不可少的一部分,它有助于提高代码的可读性和维护性,:本文主要介绍Python中注释使用方法的相关资料,需要的朋友可以参考下... 目录一、前言二、什么是注释?示例:三、单行注释语法:以 China编程# 开头,后面的内容为注释内容示例:示例:四

Redis出现中文乱码的问题及解决

《Redis出现中文乱码的问题及解决》:本文主要介绍Redis出现中文乱码的问题及解决,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1. 问题的产生2China编程. 问题的解决redihttp://www.chinasem.cns数据进制问题的解决中文乱码问题解决总结

一文详解Git中分支本地和远程删除的方法

《一文详解Git中分支本地和远程删除的方法》在使用Git进行版本控制的过程中,我们会创建多个分支来进行不同功能的开发,这就容易涉及到如何正确地删除本地分支和远程分支,下面我们就来看看相关的实现方法吧... 目录技术背景实现步骤删除本地分支删除远程www.chinasem.cn分支同步删除信息到其他机器示例步骤

全面解析MySQL索引长度限制问题与解决方案

《全面解析MySQL索引长度限制问题与解决方案》MySQL对索引长度设限是为了保持高效的数据检索性能,这个限制不是MySQL的缺陷,而是数据库设计中的权衡结果,下面我们就来看看如何解决这一问题吧... 目录引言:为什么会有索引键长度问题?一、问题根源深度解析mysql索引长度限制原理实际场景示例二、五大解决

Springboot如何正确使用AOP问题

《Springboot如何正确使用AOP问题》:本文主要介绍Springboot如何正确使用AOP问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录​一、AOP概念二、切点表达式​execution表达式案例三、AOP通知四、springboot中使用AOP导出

在Golang中实现定时任务的几种高效方法

《在Golang中实现定时任务的几种高效方法》本文将详细介绍在Golang中实现定时任务的几种高效方法,包括time包中的Ticker和Timer、第三方库cron的使用,以及基于channel和go... 目录背景介绍目的和范围预期读者文档结构概述术语表核心概念与联系故事引入核心概念解释核心概念之间的关系

在Linux终端中统计非二进制文件行数的实现方法

《在Linux终端中统计非二进制文件行数的实现方法》在Linux系统中,有时需要统计非二进制文件(如CSV、TXT文件)的行数,而不希望手动打开文件进行查看,例如,在处理大型日志文件、数据文件时,了解... 目录在linux终端中统计非二进制文件的行数技术背景实现步骤1. 使用wc命令2. 使用grep命令