前言
当我最初写游戏时,我经常使用标准Random()函数,然后写一堆if和else条件来我获得预期结果。如果结果不太好,我会写更多的条件进行过滤或者筛选,直到我觉得游戏变得有趣。最近我发现有更好的方法。内置的Random类并没有问题,问题是使用内置的Random类很难达到我们的预期效果。
现实生活中,以抛硬币为例,时而会抛出连续多次花或者字。那么如果在游戏中,可能表现为多次连续的暴击或是硬直,尽管有时暴击和硬直的概率很低,但依旧有可能连续出现,让人感觉诡异。那么为了解决类似的问题,前辈们想了很多种方法,比如说特殊值过滤等等。我们今天介绍的是Shuffle Bag技术,可以让你的随机数不那么随机,由你掌控。
什么是Shuffle Bag呢?
Shuffle Bag是一项让我们可以控制随机数分布的技术。
其主要原理为:
- 设计一个特定分布数值的集合。
- 把集合中数值转载至背包中。
- 将背包中数值随机打乱。
- 从背包中以任意顺序(从前往后,从后往前都无所谓)一个一个的抽取数值,抽完不放回,直到背包为空。
- 背包为空后,将数据放回,并重新打乱,之后循环利用。
不难看出,Shuffle Bag 特性如下:
- 不会产生集合之外的数据。
- 它仍然具有随机性,元素出现的顺序还是随机的。
- 非重复抽取,如果数据集中某个元素只有一个,至多连续出现两次。
如何实现Shuffle Bag?
原文使用C#,这里使用Unity C#和泛型,大家可以很方便的转译为其他语言。实现并不是先把整个背包先随机打乱,而是每次抽取时,才进行一次随机交换,这样做可以分摊性能,不至于在背包数据较多时,打乱背包消耗某帧过多性能。
using System.Collections.Generic;
using UnityEngine;
public class ShuffleBag<T> {
private List<T> data;
private T currentItem;
private int currentPosition = -1;
private int Capacity { get { return data.Capacity; } }
public int Size { get { return data.Count; } }
public ShuffleBag(int initCapacity)
{
data = new List<T>(initCapacity);
}
public void Add(T item, int amount)
{
for (int i = 0; i < amount; i++)
data.Add(item);
currentPosition = Size - 1;
}
public T Next()
{
if (currentPosition < 1)
{
currentPosition = Size - 1;
currentItem = data[0];
return currentItem;
}
var pos = Random.Range(0,currentPosition);
currentItem = data[pos];
data[pos] = data[currentPosition];
data[currentPosition] = currentItem;
currentPosition--;
return currentItem;
}
}
如何使用Shuffle Bag?
在我的项目中,我希望NPC播放受伤动画的概率为十分之一,并且我不希望看到他偶尔连续播放受伤动画多次,也不想长时间不播放受伤动画。
ShuffleBag<bool> hurtBag = new ShuffleBag<bool>(10);
hurtBag.Add(false, 9);
hurtBag.Add(true, 1);
//当触发受伤时,调用以下逻辑
if(hurtBag.Next())
{
Play();
}
附录
附上参考的原文链接 Shuffle Bags: Making Random() Feel More Random 英文好的同学可以直接看原文 。另外,这个原文题目有些让人疑惑,所以我的译文标题做了一些更改。