lua实现游戏抽奖的几种方法
^_^内容原创,禁止转载
假设配置如下:
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实现游戏抽奖的几种方法的更多相关文章
- 【转载】解决繁体、日文游戏乱码的五种方法 转载自:http://tieba.baidu.com/p/488627981
方法1:转换区域 开始——设置——控制面板——区域和语言选项——分别选择“高级”和“区域选项”标签——在其下拉框中都选择“日语”(或“日本”)(选项有点多,慢慢找)——重启后即可生效. *某影注:日语 ...
- Unity3d获取游戏对象的几种方法
1.GameObject.Find() 通过场景里面的名子或者一个路径直接获取游戏对象. GameObject root = GameObject.Find("GameObject" ...
- 游戏对象消失三种方法的区别?(enabled/Destroy/active)
gameObject.renderer.enabled=fasle是控制一个物体是否在屏幕上渲染或显示 而物体实际还是存在的 只是想当于隐身 而物体本身的碰撞体还依然存在的GameObject.De ...
- slua中,绑定lua文件到Monobehavior的一种方法
slua本身并不提供如何把一个lua文件绑定到一个预制中,就像一个普通的继承自monobehavior的自定义脚本那样,而tolua的框架却采用了拙劣的做法: public class LuaBeha ...
- unity3d 游戏对象消失三种方法的区别(enabled/Destroy/active)
gameObject.renderer.enabled //是控制一个物体是否在屏幕上渲染或显示 而物体实际还是存在的 只是想当于隐身 而物体本身的碰撞体还依然存在的 GameObject.Destr ...
- 【Cocos2d-x游戏开发】解决Cocos2d-x中文乱码的三种方法
众所周知,Cocos2d-x是一款不错的开源引擎,但是在Cocos2d-x中直接使用中文是无法正确显示的.比如下面的情况: 解决这个问题常用的有三种方法:1.通过转换为UTF-8编码来显示.2.使用i ...
- C模块回调Lua函数的两种方法
作者:ani_di 版权所有,转载务必保留此链接 http://blog.csdn.net/ani_di C模块回调Lua函数的两种方法 lua和C通过虚拟栈这种交互方式简单而又可靠,缺点就是C做栈平 ...
- [Unity3D]Unity3D游戏开发Lua随着游戏的债券(于)
---------------------------------------------------------------------------------------------------- ...
- lua中 table 重构index/pairs元方法优化table内存占用
转载请标明出处http://www.cnblogs.com/zblade/ lua作为游戏的热更新首选的脚本,其优势不再过多的赘述.今天,我主要写一下如何重写lua中的元方法,通过自己的重写来实现对l ...
随机推荐
- IOS 常用的宏定义(#define)
开发中经常用到的常量定义(随时更行): 与UIView相关 //获取View的frame属性 #define GetViewWidth(view) view.frame.size.width #def ...
- React Hook挖坑
React Hook挖坑 如果已经使用过 Hook,相信你一定回不去了,这种用函数的方式去编写有状态组件简直太爽啦. 如果还没使用过 Hook,那你要赶紧升级你的 React(v16.8+),投入 H ...
- Asp.Net.Core WebApi 版本控制
前言 在后端Api的开发过程中,无法避免的会遇到接口迭代的过程,如何保证新老接口的共存和接口的向前的兼容呢,这时候就需要对Api进行版本的控制,那如何优雅的控制Api的版本呢? 开始 Microsof ...
- java web数据库的增删改查详细
本次课上实验是完成数据库的增删改查. 包括增加用户信息.删除用户信息.多条件查找用户信息.修改用户信息(主要是复选框单选框等的相关操作.) 下面下看一下各个界面的样子. 总页面:显示全部页面:增加页面 ...
- 100 Path Sum
Given a binary tree and a sum, determine if the tree has a root-to-leaf path such that adding up all ...
- 计算机网络协议,PPP协议分析
一.基本特点 1.PPP协议是计算机网络体系中第二层(数据链路层)的协议 2.PPP帧格式是以HDLC帧格式为基础,做了很少的改动(区别:PPP是面向字符的,而HDLC是面向位的) 3.PPP协议使用 ...
- hive常用函数四
字符串函数 1. 字符串长度函数:length 语法: length(string A) 返回值: int 说明:返回字符串A的长度 举例: hive> select length('abced ...
- Centos7_Root密码重置
原因: 最近出去见女朋友,竟然忘了Root用户的密码,此时考验linux基础扎不扎实的时候到了... 操作步骤: 解释补充: mount -o remountr,w / #修改根目录文件系统的权限,实 ...
- .NET Core技术研究-主机
前一段时间,和大家分享了 ASP.NET Core技术研究-探秘Host主机启动过程 但是没有深入说明主机的设计.今天整理了一下主机的一些知识,结合先前的博文,完整地介绍一下.NET Core的主机的 ...
- 【python系统学习14】类的继承与创新
目录: 目录: [toc] 类的继承 子类和父类 继承的写法 继承示例 父类可以被无限个子类所继承 子类实例可调用父类属性和方法 类的始祖(根类) 根类 - object 实例归属判断 - isins ...