游戏中的伪随机机制

发表于2015-07-20
评论5 1.37w浏览

                                                       

本文首发GAD-腾讯游戏开发者平台(http://gad.qq.com),转载请获得原作者授权并标明出处。

游戏中的伪随机机制

游戏中,存在着大量随机事件比如常见的暴击游戏中的抽奖对于随机事件的概率,我们通对认为随机事件每一次发生的概率等于设定的概率,从另一个角度来说,随机事件每一次发生的概率都固定且不受事件上一次结果的影响。大量样本计算的情况下,并不存在什么问题。然而实际游戏过程中,我们会更关注相对小的样本下,也就是有限次操作(攻击/抽奖过程中,随机事件带来感受为了保证有限次操作的概率相对稳定,营造良好的玩家体验,引入了伪随机的机制。本文对游戏中常见的随机过程做简要介绍,对伪随机过程较为详细的分析,并给出了一种简单的由概率求取伪随机过程的方法。

游戏中的随机概率一般可以分为两类,一类是固定的概率,另一类是根据游戏行为有变化的概率。

比如20%的暴击率,我们通常认为每一次攻击触发暴击都是固定的几率20%,从另一个角度来说,每一次攻击暴击率独立且固定的并不受攻击次数和之前攻击效果的影响实际过程中,玩家有了另外一种理解或者说是期待,5次攻击出现1次暴击更宽泛的一点说,玩家会认为在3~7(一般4~6为佳的过程中发生一次暴击。对于游戏策划的设计来说,一般并不希望玩家能够在20%的暴击率下,获得较高概率的连续两次暴击,同时也不希望会有连续10次不发生暴击的情况。这些问题传统固定概率的情况下较难得到解决。

对于固定的概率,定义P(K)为K-1未发生暴击,第K发生暴击的概率,则P(K=N):

这是一个单调递减函数,随着N的不断扩大,P不断减小,在有限次操作的状态下,玩家对于20%容易缺少一个较为准确的感知。为了增强这种感知,可以让P在K=5附近拥有较大的概率,在其他情况下拥有较小的概率。当然在K=5之外,P趋近于0的速度是我们另外一个可以关注的问题,在此暂不做讨论。

 

下表为通过固定概率和伪随机过程获得暴击率的对比,P(K)为前K-1未发生暴击,第K次发生暴击的概率,S(K)为前K项之和。

K

P(K)_固定

S(K)_固定

P(K)_伪随机

S(K)_伪随机

1

0.2

0.2

0.0557

0.0557

2

0.16

0.36

0.10519502

0.16089502

3

0.128

0.488

0.140214442

0.301109462

4

0.1024

0.5904

0.155712812

0.456822274

5

0.08192

0.67232

0.151274997

0.608097271

6

0.065536

0.737856

0.130973892

0.739071163

7

0.0524288

0.7902848

0.101736154

0.840807316

8

0.04194304

0.83222784

0.07093626

0.911743576

9

0.033554432

0.865782272

0.044242945

0.955986521

10

0.026843546

0.892625818

0.024515508

0.980502029

11

0.021474836

0.914100654

0.011946407

0.992448436

12

0.017179869

0.931280523

0.005047465

0.997495901

13

0.013743895

0.945024419

0.001813218

0.999309119

14

0.010995116

0.956019535

0.000538749

0.999847868

15

0.008796093

0.964815628

0.000127106

0.999974974

16

0.007036874

0.971852502

2.23029E-05

0.999997277

17

0.0056295

0.977482002

2.57822E-06

0.999999855

下图更直观的展示了这种差别。

从图中,我们可以非常清晰的看到通过伪随机函数能够较好实现预期对于图表的进一步分析,在K=5附近的概率,3<K<7的概率,以及方差,在此暂不做讨论

接下来我们对于伪随机过程做介绍,我们继续以暴击率为例。

对于暴击率来说,伪随机机制通过不同攻击次数拥有不同的实际暴击率来实现,具体规则如下

设定初始的概率为P,定义随机种子C一个常数目前对于DOTA剑圣的暴击率分析较为全面,本文也将以剑圣为例介绍伪随机机制,所以设定P=20%,C=5.57%。

1. 进行第一次攻击,暴击概率为C。如果未暴击,进入流程2;如果暴击返回流程1进行下次暴击计算。

2. 进行第二次攻击,暴击率为2C。如果未暴击,进入流程3如果暴击返回流程1进行下次暴击计算。

3.进行第三次攻击,暴击率为3C。如果未暴击,进入流程4如果暴击返回流程1进行下次暴击计算。

。。。。

17. 进行第十七次攻击,暴击率为17C。如果未暴击,进入流程18如果暴击返回流程1进行下次暴击计算。

18. 本次攻击暴击。返回1进行下次暴击计算。

也就是说,如果一个英雄始终不暴击,那么他的暴击概率会是:C2C3C4C...NC。当N足够大时,NC会大于等于1,所以一定会发生暴击。实际上,英雄攻击的暴击概率始终在若干个不同暴率的状态中切换。但是,当这种攻击进行次数很多以后,此英雄的平均暴击概率会趋近于20%

下面表格是从网上搜索到WAR3关于显示概率P、随机种子C、暴击前攻击的最大次数 maxN实际统计概率Pcount数据表格。

P

C

maxN

Pcount

5.00%

0.0038

263

5.00%

10.00%

0.01475

67

10.00%

15.00%

0.03221

31

15.00%

20.00%

0.0557

17

20.00%

25.00%

0.08475

11

24.90%

30.00%

0.11895

8

29.10%

35.00%

0.14628

6

33.60%

40.00%

0.18128

5

37.80%

45.00%

0.21867

4

41.60%

50.00%

0.25701

3

45.70%

55.00%

0.29509

3

49.40%

60.00%

0.33324

3

53.00%

65.00%

0.38109

2

56.40%

70.00%

0.42448

2

60.10%

75.00%

0.46134

2

63.20%

80.00%

0.50276

1

66.70%

85.00%

0.5791

1

70.30%

90.00%

0.67068

1

75.00%

95.00%

0.77041

1

81.30%

 

接下来通过C值计算P值过程做介绍

在此我们

K次攻击的暴击率(之前K-1未发生暴击)

代表K次成功暴击的概率,即前K-1未发生暴击,第K次发生暴击。

综上,成功暴击一次所需要的平均次数为

所以暴击的实际概率为:

通过以上推导,我们可以根据C值求出实际概率Pa

对于我们在实际使用过程中,我们关注如何通过P值求得C值P和c的函数关系单调函数,在此我们介绍一种简单的方法进行实现,最小二分法在最优化理论中有很多可以减少迭代次数的方法,欢迎大家进行尝试。

最小二分法如下:

首先我们会定义一些参数,C1,C2,C3eps初始化C1=0,C2=1暴击率精度eps=0.01%预设暴击率为Ps

1.C3=C1+C2/2

2.计算C3的实际Pa。如果PaPs差值绝对值小于eps,则结束计算,C3即为所需要的种子如果PaPs差值大于eps进入流程3。

3.如果Pa<Ps,则C1=C3;如果Pa>Ps,则C2=C3。返回流程1

    通过有限次的迭代,我们就可以获得随机种子C。

通过以上的伪随机机制,有利于游戏中概率事件的稳定。对于暴击事件来说,伪随机机制降低了连续暴击或者连续不暴击的的概率,使得玩家的体验更为友好。

现在手机游戏中,广泛采用 了可以保证一定收益的抽奖方法,比如玩家的十连抽,这种过程也是伪随机过程的一种。这种十连抽必中的策略,保证了玩家拥有较为平稳的体验,也有利于游戏体验节奏的把控。对于这些伪随机过程在此不一一分析。对于上文提到的如果使概率更为集中的方法,有兴趣的读者也可以进一步研究。

本文使用了网上的一些资料,大家表示感谢,希望本文能够为入门同学提供一些帮助。如果大家对于复杂的随机过程感兴趣,欢迎一起讨论。如有版权问题,请联系我。

 

如社区发表内容存在侵权行为,您可以点击这里查看侵权投诉指引