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 ...
随机推荐
- Docker搭建MySQL主从复制
Docker搭建MySQL主从复制 主从服务器上分别安装Docker 1.1 Docker 要求 CentOS 系统的内核版本高于 3.10 [root@localhost ~]# uname -r ...
- Win10下 VSCode配置LaTex环境
------------恢复内容开始------------ 环境:win10 软件:Tex Live, VSCode ,Sumatra Tex Live官网下载速度很慢,所以推荐使用清华镜像下载链接 ...
- JSP学习笔记(三)
jsp与javabean 编写javabean 创建与使用bean 获取和修改bean的属性 javabean是一种java类,是通过封装属性和方法成为具有某种功能或处理某个业务的对象,简称bean, ...
- 1.用eclipse创建maven工程
第一步.File→New→Maven Project (需要下载安装配置Maven等,这些步骤省略) (找不到的话选Other,里面的Maven文件夹里有) 二.记得勾选上,然后点Next 三.填完点 ...
- 微服务架构盛行的时代,你需要了解点 Spring Boot
随着互联网的高速发展,庞大的用户群体和快速的需求变化已经成为了传统架构的痛点. 在这种情况下,如何从系统架构的角度出发,构建出灵活.易扩展的系统来快速响应需求的变化,同时,随着用户量的增加,如何保证系 ...
- JavaScript new 的时候到底发生了什么?
function Person(name) { this.name = name; } let liLei = new Person('lilei'); console.log(liLiei.name ...
- 让你第一次认识到Java的内存管理
发现之前写的可读性不好,这次准备试试换风格,去掉长篇大论,觉得这个风格好的,麻烦点个赞啦 清理.JVM的妙处 大家以后都是程序员,假设你很不幸,需要自己交钱租房子. 你作为一个小穷人,租的房子到期了( ...
- python实现杨辉三角形
代码实现: # python实现杨辉三角形 def yanghui(): # 定义第一行列表为[1] line = [1] while True: # yield的作用:把一个函数变成生成器,同时返回 ...
- VM虚拟机复制文件问题
需要安装好vmtools,安装好后,启动虚拟机环境: 把需要复制的文件拖进虚拟机环境窗口,鼠标指针会变成复制图标,直接左键即可复制: 不能Ctrl+c-Ctrl+v进去.
- VUE CLI3.0安装及配置
# 安装 npm install -g @vue/cli # 查看已安装版本vue --version 或者 vue -V # 卸载 npm uninstall @vue/cli -g # 新建项目 ...