详谈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基本编码模式小结
May 23 Javascript
关于JS管理作用域的问题
Apr 10 Javascript
点击弹出层效果&amp;弹出窗口后网页背景变暗效果的实现代码
Feb 10 Javascript
谈谈我对JavaScript原型和闭包系列理解(随手笔记9)
Dec 24 Javascript
Vue中fragment.js使用方法详解
Mar 09 Javascript
jQuery实现页面倒计时并刷新效果
Mar 13 Javascript
详解vue项目首页加载速度优化
Oct 18 Javascript
基于VUE实现的九宫格抽奖功能
Sep 30 Javascript
uni-app之APP和小程序微信授权方法
May 09 Javascript
Vue Components 数字键盘的实现
Sep 18 Javascript
Vue实现浏览器打印功能的代码
Apr 17 Javascript
vue + node如何通过一个Txt文件批量生成MP3并压缩成Zip
Jun 02 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 生成随机验证码图片代码
2010/02/08 PHP
php获取qq用户昵称和在线状态(实例分析)
2013/10/27 PHP
PHP正则表达式 /i, /is, /s, /isU等介绍
2014/10/23 PHP
php实现utf-8转unicode函数分享
2015/01/06 PHP
ThinkPHP数据操作方法总结
2015/09/28 PHP
PHP连接数据库实现注册页面的增删改查操作
2016/03/27 PHP
CI框架中数据库操作函数$this-&gt;db-&gt;where()相关用法总结
2016/05/17 PHP
Laravel中日期时间处理包Carbon的简单使用
2017/09/21 PHP
javascript面向对象编程(一) 实例代码
2010/06/25 Javascript
jQuery.extend()、jQuery.fn.extend()扩展方法示例详解
2014/05/08 Javascript
兼容Firefox的Javascript XSLT 处理XML文件
2014/12/31 Javascript
js实现表格筛选功能
2017/01/18 Javascript
Vue2.0 组件传值通讯的示例代码
2017/08/01 Javascript
通过jquery的ajax请求本地的json文件方法
2018/08/08 jQuery
js中的深浅拷贝问题简析
2019/05/10 Javascript
JavaScript实现五子棋游戏的方法详解
2019/07/08 Javascript
浅谈vue项目用到的mock数据接口的两种方式
2019/10/09 Javascript
Vue实现按钮级权限方案
2019/11/21 Javascript
微信小程序中的列表切换功能实例代码详解
2020/06/09 Javascript
[41:52]2018DOTA2亚洲邀请赛3月29日 小组赛A组 TNC VS OpTic
2018/03/30 DOTA
python正则表达式re模块详细介绍
2014/05/29 Python
Python 基础教程之包和类的用法
2017/02/23 Python
Python和C/C++交互的几种方法总结
2017/05/11 Python
对Python3之进程池与回调函数的实例详解
2019/01/22 Python
django formset实现数据表的批量操作的示例代码
2019/12/06 Python
python实现在内存中读写str和二进制数据代码
2020/04/24 Python
python的reverse函数翻转结果为None的问题
2020/05/11 Python
巴西食品补充剂在线零售商:Músculos na Web
2017/08/07 全球购物
德国黑胶唱片、街头服装及运动鞋网上商店:HHV
2018/08/24 全球购物
俄罗斯珠宝市场的领导者之一:Бронницкий ювелир
2019/10/02 全球购物
普通院校学生的自荐信
2013/11/27 职场文书
大学新生军训自我鉴定
2014/03/18 职场文书
党员教师个人对照检查材料(群众路线)
2014/09/26 职场文书
廉政承诺书范文
2015/04/28 职场文书
CSS预处理框架——Stylus
2021/04/21 HTML / CSS
MySQL 如何设计统计数据表
2021/06/15 MySQL