^_^内容原创,禁止转载


 假设配置如下:

 local reward_pool = {
     {weight = , item = {, num = }},
     {weight = , item = {, num = }},
     {weight = , item = {, num = }},
     {weight = , item = {, num = }},
     {weight = , item = {, num = }},
     {weight = , item = {, num = }},
 }

1.顺序查找,预处理时间复杂度O(n),抽奖最坏情况O(n)

 --预处理
 local N = #reward_pool

 for _, v in ipairs(reward_pool) do
     total_weight = total_weight + v.weight
 end

 --实现
 local rand_weight = math.random(total_weight)
 local reward_index

 for k, v in ipairs(reward_pool) do
     _total_weight = _total_weight + v.weight
     if _total_weight >= rand_weight then
         reward_index = k
         break
     end
 end

2.按照离散思路进行分割,二分查找,预处理时间复杂度O(n),抽奖最坏情况O(logn)

 --预处理
 local N = #reward_pool

 for _, v in ipairs(reward_pool) do
     com_weight = com_weight + v.weight
     v.weight = com_weight
 end

 --实现
 , #reward_pool
 while right >= left then
     )
     local mid_weight = reward_pool[mid].weight
     if value == mid_weight then
         right = right -
         break
     elseif value < mid_weight then
         right = mid -
     else
         left = mid +
     end
 end
 right = right +  --此时right为reward_pool中抽到的索引

这种方法在实际上是对第一种方法的优化,在大多数情况下都可以取代第一种方法,但取舍还要看实际情况,一个极端且明显的例子如下:

 local reward_pool = {
     {weight = , item = {, num = }}, {weight = , item = {, num = }},
     {weight = , item = {, num = }}, {weight = , item = {, num = }},
     {weight = , item = {, num = }}, {weight = , item = {, num = }},
     {weight = , item = {, num = }}, {weight = , item = {, num = }},
     {weight = , item = {, num = }}, {weight = , item = {, num = }},
 }

3.AliasMethod,个人实现的预处理O(3n),抽奖时间复杂度O(1),下面是实现过程,证明日后有时间再整理给出

 queue = {}

 function queue:new()
     , last = -}
     self.__index = self
     setmetatable(res, self)
     return res
 end

 function queue:push(value)
     self.last = self.last +
     self[self.last] = value
 end

 function queue:pop()
     local first = self.first
     if first > self.last then
         self.first =
         self.last = -
         return nil
     end
     local value = self[first]
     self[first] = nil
     self.first = self.first +
     return value
 end

 function queue:front()
     return self[self.first]
 end

 --预处理
 local N = #reward_pool

 for _, v in ipairs(reward_pool) do
     total_weight = total_weight + v.weight
 end

 local Prob = {}
 local Alias = {}
 local weightN_queue_L = queue:new()
 local weightN_queue_U = queue:new()
 for k, v in ipairs(reward_pool) do
     local weight_N = v.weight * N
     if weight_N == total_weight then
         Prob[k] = weight_N
     else
         local tb = {index = k, value = weight_N}
         local qu = weight_N > total_weight and weightN_queue_U or weightN_queue_L
         qu:push(tb)
     end
 end

 while true do
     local l_qu = weightN_queue_L:pop()
     if not l_qu then
         break
     end
     local u_qu = weightN_queue_U:front() --或直接pop,比total_weight大再push回去
     Prob[l_qu.index] = l_qu.value
     Alias[l_qu.index] = u_qu.index
     u_qu.value = u_qu.value + l_qu.value - total_weight
     if u_qu.value < total_weight then
         weightN_queue_U:pop()
         weightN_queue_L:push(u_qu)
     elseif u_qu.value == total_weight then
         weightN_queue_U:pop()
         Prob[u_qu.index] = total_weight
     end
 end
 weightN_queue_U = nil
 weightN_queue_L = nil

 --实现
 local n = math.random(N)
 local weight = math.random(total_weight)
 local reward_index = weight > Prob[n] and Alias[n] or n

lua实现游戏抽奖的几种方法的更多相关文章

  1. 【转载】解决繁体、日文游戏乱码的五种方法 转载自:http://tieba.baidu.com/p/488627981

    方法1:转换区域 开始——设置——控制面板——区域和语言选项——分别选择“高级”和“区域选项”标签——在其下拉框中都选择“日语”(或“日本”)(选项有点多,慢慢找)——重启后即可生效. *某影注:日语 ...

  2. Unity3d获取游戏对象的几种方法

    1.GameObject.Find() 通过场景里面的名子或者一个路径直接获取游戏对象. GameObject root = GameObject.Find("GameObject" ...

  3. 游戏对象消失三种方法的区别?(enabled/Destroy/active)

    gameObject.renderer.enabled=fasle是控制一个物体是否在屏幕上渲染或显示  而物体实际还是存在的 只是想当于隐身 而物体本身的碰撞体还依然存在的GameObject.De ...

  4. slua中,绑定lua文件到Monobehavior的一种方法

    slua本身并不提供如何把一个lua文件绑定到一个预制中,就像一个普通的继承自monobehavior的自定义脚本那样,而tolua的框架却采用了拙劣的做法: public class LuaBeha ...

  5. unity3d 游戏对象消失三种方法的区别(enabled/Destroy/active)

    gameObject.renderer.enabled //是控制一个物体是否在屏幕上渲染或显示 而物体实际还是存在的 只是想当于隐身 而物体本身的碰撞体还依然存在的 GameObject.Destr ...

  6. 【Cocos2d-x游戏开发】解决Cocos2d-x中文乱码的三种方法

    众所周知,Cocos2d-x是一款不错的开源引擎,但是在Cocos2d-x中直接使用中文是无法正确显示的.比如下面的情况: 解决这个问题常用的有三种方法:1.通过转换为UTF-8编码来显示.2.使用i ...

  7. C模块回调Lua函数的两种方法

    作者:ani_di 版权所有,转载务必保留此链接 http://blog.csdn.net/ani_di C模块回调Lua函数的两种方法 lua和C通过虚拟栈这种交互方式简单而又可靠,缺点就是C做栈平 ...

  8. [Unity3D]Unity3D游戏开发Lua随着游戏的债券(于)

    ---------------------------------------------------------------------------------------------------- ...

  9. lua中 table 重构index/pairs元方法优化table内存占用

    转载请标明出处http://www.cnblogs.com/zblade/ lua作为游戏的热更新首选的脚本,其优势不再过多的赘述.今天,我主要写一下如何重写lua中的元方法,通过自己的重写来实现对l ...

随机推荐

  1. Codeforces #614 div.2 (A-E)

    A  ConneR and the A.R.C. Markland-N #include <bits/stdc++.h> using namespace std; #define ll l ...

  2. Unity引擎入门——制作第一个2D游戏(1)

    Unity作为当今最流行的游戏引擎之一,受到各大厂商的喜爱. 像是炉石传说,以及最近的逃离塔克夫,都是由unity引擎开发制作. 作为初学者的我们,虽然无法直接做出完成度那么高的作品,但每一个伟大的目 ...

  3. Django-使用 include() 配置 URL

    如果项目非常庞大,应用非常多,应用的 URL 都写在根 urls.py 配置文件中的话,会显的非常杂乱,还会出现名称冲突之类的问题,这样对开发整个项目是非常不利的. 可以这样解决,把每个应用的 URL ...

  4. java中eclipse的安装和JDK的环境变量的配置以及记事本的使用

    2020-04-09 23:26:15 学习java的第一步当然就是环境配置了,java中的配置作为小白刚刚开始肯定会有点一点晕头转向的,开没等开始入门呢!就要准备放弃了.哈哈哈哈,没关系的,都是这么 ...

  5. 1000行MySQL学习笔记,不怕你不会,就怕你不学!

    Windows服务 -- 启动MySQL net start mysql-- 创建Windows服务 sc create mysql binPath= mysqld_bin_path(注意:等号与值之

  6. python函数的传参模式

    python里的变量更像是一个名字.标签.而Python中一切又皆为对象. 当函数传参时,函数参数作为一个标签,指向某个对象,因此更贴切的说是"call by object". 但 ...

  7. (C++ Error: Incompatible types in assignment of ‘char*’ to ‘char [2])

    in C++, cannot assign a pointer to an array. c++中char*与char[]不是一种类型,但是在C里面可以,所以尽量使用C++中的类,如string, v ...

  8. Linux 文件管理篇(四 文件查找)

    显示脚本文件或者命令的路径                                    which 显示档案的路径(数据库中查找,较快)                           ...

  9. 【Selenium01篇】python+selenium实现Web自动化:搭建环境,Selenium原理,定位元素以及浏览器常规操作!

    一.前言 最近问我自动化的人确实有点多,个人突发奇想:想从0开始讲解python+selenium实现Web自动化测试,请关注博客持续更新! 二.话不多说,直接开干,开始搭建自动化测试环境 这里以前在 ...

  10. 31.2 try finally使用

    package day31_exception; import java.io.FileWriter; import java.io.IOException; import java.lang.Exc ...