设为首页收藏本站|繁體中文

Project1

 找回密码
 注册会员
搜索
123
返回列表 发新帖
楼主: 北意大利
打印 上一主题 下一主题

[有事请教] 如何实现随机抽到的数字不重复(7月31日追加问题)

[复制链接]

Lv3.寻梦者

梦石
0
星屑
1221
在线时间
144 小时
注册时间
2020-4-26
帖子
148
21
发表于 2020-8-1 02:13:13 | 只看该作者
本帖最后由 moonyoulove 于 2020-8-1 02:38 编辑

先初始化签筒,把签筒保存全局变量或rm的变量里,
再来决定要抽取几道题目,最大值应为6*6*6(依照配置)
参数传入签筒(types),重复执行randomCode,每次都能得到随机不重复的题目编号了
JAVASCRIPT 代码复制
  1. //初始化题库的签筒
  2.       function initQuestion() {
  3.         var typeNum = 6; //类型数量
  4.         var hardNum = 6; //难度数量
  5.         var questNum = 6; //序号数量
  6.         // 建立一个多重嵌套的数组(多维阵列),如下
  7.         //类型1─难度1─序号1
  8.         //   │     └序号2
  9.         //   │     └序号3
  10.         //   └难度2─序号1
  11.         //        └序号2
  12.         var types = [];
  13.         for (var i = 0; i < typeNum; i++) {
  14.           var hards = [];
  15.           for (var j = 0; j < hardNum; j++) {
  16.             var quests = [];
  17.             for (var k = 0; k < questNum; k++) {
  18.               //存放一个对象,其里面的num代表实际序号
  19.               quests.push({
  20.                         num: k + 1
  21.                       });
  22.             }
  23.             //除了num,还得存放该难度下的序号数组
  24.             hards.push({
  25.               num: j + 1,
  26.               quests: quests,
  27.             });
  28.           }
  29.           //同理,只是改放难度数组
  30.           types.push({
  31.             num: i + 1,
  32.             hards: hards,
  33.           });
  34.         }
  35.         //全部数组建好后,返回types
  36.         return types;
  37.       }
  38.  
  39.       //抽签,原理是对建立好的签筒进行抽完不放回的动作
  40.       function randomCode(types) {
  41.                 //先抽类型,i、j、k会依序存放抽到的是第几支签
  42.         var i = rand(types.length); //表示随机types数组的到第一位到最后一位
  43.         var type = types[i]; // type实际长这样{num:第几个类型, hards:类型里的难度数组}
  44.         var hards = type.hards; //取得难度数组
  45.         var j = rand(hards.length);//第二阶段,抽难度
  46.         var hard = hards[j];
  47.         var quests = hard.quests;
  48.         var k = rand(quests.length);//第三阶段,抽题目序号
  49.         var quest = quests[k];
  50.         //已经得到一组类型、难度和序号
  51.         //实际数值是放在num里
  52.         //code可以依照需求改写,下面是用「-」连成字符串
  53.         var code = type.num + "-" + hard.num + "-" + quest.num;
  54.         //等三支签(类型、难度、序号)抽完,再进行丢掉的动作
  55.         quests[k] = quests[quests.length - 1]; //把最后一个元素覆盖到抽到的元素的位置
  56.         quests.pop();//舍弃掉最后的元素
  57.         //抽出来的序号签已经丢掉了,接下来换难度的签
  58.         //于是这里检查这个难度底下是不是还有题目,没有的话把这个难度从数组删掉
  59.         if (quests.length == 0) {
  60.                   //跟上面同样原理
  61.           hards[j] = hards[hards.length - 1];
  62.           hards.pop();
  63.         }
  64.         //再来换检查是否要从整个数组删除已经不需要的种类
  65.         if (hards.length == 0) {
  66.           types[i] = types[types.length - 1];
  67.           types.pop();
  68.         }
  69.         //最后返回处理好的编码,此处能做变通,比如返回数组,或不返回,直接赋值给rm的变量
  70.         return code;
  71.  
  72.         //内部的函数,用来方便地得到一个从0到i,但不包括i的随机整数
  73.         function rand(i) {
  74.           return Math.floor(Math.random() * i);
  75.         }
  76.       }

点评

注解好详细!!辛苦了!!这里学到了很多、问题也解决了><实在是非常感谢!!  发表于 2020-8-1 14:59
我发现这种方法适合在题目少,或者抽取的数目接近总数的时候,如果是100万道题抽10题,那签可以抽完后放回,因为再抽到的机率低,再次重抽即可  发表于 2020-8-1 02:49

评分

参与人数 1+1 收起 理由
北意大利 + 1 精品文章

查看全部评分

回复 支持 2 反对 0

使用道具 举报

您需要登录后才可以回帖 登录 | 注册会员

本版积分规则

拿上你的纸笔,建造一个属于你的梦想世界,加入吧。
 注册会员
找回密码

站长信箱:[email protected]|手机版|小黑屋|无图版|Project1游戏制作

GMT+8, 2024-4-29 03:11

Powered by Discuz! X3.1

© 2001-2013 Comsenz Inc.

快速回复 返回顶部 返回列表