详谈JS中实现种子随机数及作用


Posted in Javascript onJuly 19, 2016

前言

在前端开发中,尤其是游戏开发,经常会用到随机数,那么我们会第一时间想到:Math.random,大家略微的看看如下代码:

for (var i= 0; i<10; i++) { document.writeln(Math.random() +"<br />"); }

运行如上代码,也确实生成了10个不同的数字,当然你可以生成更多,看起来挺不错的,如果仅仅如此,那么本文就没必要写了。

示例

试着想一下,如果在某一个场景,我们做一个游戏,用户玩到一半的时候退出了,这样用户下次进来可以选择继续上一次的进度继续玩,那么现在问题来了:用户玩的进度以及用户的积分等简单的描述数据,我们都可以记录下来,但是游戏里绘制的障碍物、飞行物以及很多装饰类的小玩意儿,他们甚至是每次用户点开始随机输出的,要把画布上所有的东西以及它们的大小,位置等都记录下来,实在是没必要。

于是种子随机数就闪亮登场了,我们如果在画布上元素随机绘制的时候,有一个种子值,页面上所有元素的位置、大小等都是根据这个种子来算的,那么等到第二次绘制的时候只需要传入这个种子,就可以重现之前未完成的画布元素。

那么这个时候,你会发现JS里面自带的Math.random就不好使了,无法满足需求,我们继续看这段代码:

Math.seed = 5; Math.seededRandom = function(max, min) { max = max || 1; min = min || 0; Math.seed = (Math.seed * 9301 + 49297) % 233280; var rnd = Math.seed / 233280.0; return min + rnd * (max - min); }; for (var i= 0; i<10; i++) { document.writeln(Math.seededRandom() +"<br />"); }

运行如上代码你会发现如果种子Math.seed不变,那么生成的随机数是不会变化的,哦了,如果引入这个函数,那么重现游戏场景可以实现了,虽然还需要做更多的细节处理,但机制上是能保证的,本文的重点不是实现一个这样的游戏。

本文的重点是:(Math.seed * 9301 + 49297) % 233280,为什么会是这三个值,而不是其它的到底这三个数字有什么神秘的来历呢?

像Math.seededRandom这种伪随机数生成器叫做线性同余生成器(LCG, Linear Congruential Generator),几乎所有的运行库提供的rand都是采用的LCG,形如:

I n+1=aI n+c(mod m)

生成的伪随机数序列最大周期m,范围在0到m-1之间。要达到这个最大周期,必须满足:
1.c与m互质

2.a - 1可以被m的所有质因数整除

3.如果m是4的倍数,a - 1也必须是4的倍数

以上三条被称为Hull-Dobell定理。作为一个伪随机数生成器,周期不够大是不好意思混的,所以这是要求之一。因此才有了:a=9301, c = 49297, m = 233280这组参数,以上三条全部满足。

总结

以上就是关于种子随机数在JS中如何实现和作用介绍的内容,希望给JavaScript学习者有所帮助

Javascript 相关文章推荐
JavaScript打字小游戏代码
Dec 26 Javascript
javascript之典型高阶函数应用介绍二
Jan 10 Javascript
解决checkbox的attr(checked)一直为undefined问题
Jun 16 Javascript
JavaScript实现的Tween算法及缓冲特效实例代码
Nov 03 Javascript
详解js中call与apply关键字的作用
Nov 21 Javascript
javascript 面向对象function详解及实例代码
Feb 28 Javascript
jQuery:unbind方法的使用详解
Aug 14 jQuery
vue解决跨域路由冲突问题思路解析
Nov 03 Javascript
jQuery实现的简单获取索引功能示例
Jun 04 jQuery
js中getter和setter用法实例分析
Aug 14 Javascript
js replace替换字符串同时替换多个方法
Nov 27 Javascript
浅谈layui 表单元素的选中问题
Oct 25 Javascript
全面了解JavaScript对象进阶
Jul 19 #Javascript
EasyUI中在表单提交之前进行验证
Jul 19 #Javascript
jQuery EasyUI提交表单验证
Jul 19 #Javascript
js 动态添加元素(div、li、img等)及设置属性的方法
Jul 19 #Javascript
JS把内容动态插入到DIV的实现方法
Jul 19 #Javascript
JS动态给对象添加事件的简单方法
Jul 19 #Javascript
老生常谈js动态添加事件--- 事件委托
Jul 19 #Javascript
You might like
PHP以及MYSQL日期比较方法
2012/11/29 PHP
php缩小png图片不损失透明色的解决方法
2013/12/25 PHP
详解WordPress中的头像缓存和代理中的缓存更新方法
2016/03/01 PHP
PHP7内核之Reference详解
2019/03/14 PHP
用js遍历 table的脚本
2008/07/23 Javascript
jQuery遍历页面所有CheckBox查看是否被选中的方法
2015/04/14 Javascript
jQuery的bind()方法使用详解
2015/07/15 Javascript
实现高性能JavaScript之执行与加载
2016/01/30 Javascript
【JS+CSS3】实现带预览图幻灯片效果的示例代码
2016/03/17 Javascript
js简单实现网页换肤功能
2017/04/07 Javascript
Vuex之理解Getters的用法实例
2017/04/19 Javascript
ES6学习之变量的两种命名方法示例
2017/07/18 Javascript
Vue.js实现数据响应的方法
2018/08/13 Javascript
React项目动态设置title标题的方法示例
2018/09/26 Javascript
vue多次循环操作示例
2019/02/08 Javascript
js blob类型url的视频下载问题的解决
2019/11/29 Javascript
js实现贪吃蛇游戏(简易版)
2020/09/29 Javascript
[38:31]完美世界DOTA2联赛PWL S3 Magma vs GXR 第一场 12.13
2020/12/17 DOTA
简单解决Python文件中文编码问题
2015/11/22 Python
对Python中创建进程的两种方式以及进程池详解
2019/01/14 Python
Python使用sklearn库实现的各种分类算法简单应用小结
2019/07/04 Python
PIL图像处理模块paste方法简单使用详解
2019/07/17 Python
django多对多表的创建,级联删除及手动创建第三张表
2019/07/25 Python
通过selenium抓取某东的TT购买记录并分析趋势过程解析
2019/08/15 Python
Python三元运算与lambda表达式实例解析
2019/11/30 Python
python中安装django模块的方法
2020/03/12 Python
致跳高运动员加油稿
2014/02/12 职场文书
代办委托书怎么写
2014/08/01 职场文书
2014老师三严三实对照检查材料思想汇报
2014/09/18 职场文书
2014年学校党建工作汇报材料
2014/11/02 职场文书
离职感谢信怎么写
2015/01/22 职场文书
上班迟到检讨书
2015/05/06 职场文书
2016庆祝教师节新闻稿
2015/11/25 职场文书
教师师德承诺书2016
2016/03/25 职场文书
广播稿:校园广播稿范文
2019/04/17 职场文书
原生Javascript+HTML5一步步实现拖拽排序
2021/06/12 Javascript