游戏常用算法-洗牌算法

游戏常用算法-洗牌算法洗牌算法是一个比较常见的面试题。一副扑克54张牌,有54!种排列方式。最佳的洗牌算法,应该能够等概率地生成这54!种结果中的一种

大家好,又见面了,我是你们的朋友全栈君。

洗牌算法是一个比较常见的面试题。

一副扑克54张牌,有54!种排列方式。最佳的洗牌算法,应该能够等概率地生成这54!种结果中的一种

基于Unity的洗牌算法代码实现

GitHub链接

<span role="heading" aria-level="2">游戏常用算法-洗牌算法

抽牌洗牌

原理

这是完全合乎现实洗牌逻辑的算法。

就是抽出纸牌的最后一张随机插入到牌库中,这般抽54次就完成了对扑克牌的洗牌

复杂度

空间O(1),时间O(n^2)

优缺点

如果牌库是以一个数组描述,这种插入式的洗牌不可避免地要大量移动元素。

Fisher_Yates算法

原理

取两个列表,一个是洗牌前的序列A{1,2….54),一个用来放洗牌后的序列B,B初始为空

while A不为空

随机从A取一张牌加入B末尾

复杂度

空间O(n),时间O(n^2)

代码实现

 1 List<int> list = new List<int>(pukes.pukes);//洗牌前的序列A
 2 List<int> newlist = new List<int>(list.Count);//洗牌后的序列B
 3 for(int i = 0 ; i < pukes.pukes.Length ; ++i)
 4 {
 5   int randomIndex = Random.Range(0, list.Count);
 6   int r = list[randomIndex];//随机取牌
 7    newlist.Add(r);
 8    list.RemoveAt(randomIndex);
 9 }
10 pukes.ResetPuke(newlist.ToArray());//序列B为洗牌后的结果

优缺点

算法原理清晰,但额外开辟了一个List,而且为List删除元素是不可避免地需要移动元素

通过54次生成的随机数取1/54,1/53,…1/1能等概率地生成这54!种结果中的一种

Knuth_Durstenfeld算法

Knuth 和Durstenfeld 在Fisher 等人的基础上对算法进行了改进。 每次从未处理的数据中随机取出一个数字,然后把该数字放在数组的尾部, 即数组尾部存放的是已经处理过的数字 。 这是一个原地打乱顺序的算法,算法时间复杂度也从Fisher算法的 O ( n 2 )提升到了 O ( n )。

 

1 for(int i = pukes.pukes.Length - 1;i>0;--i)
2   {
3       int randomIndex = Random.Range(0, i+1);
4       pukes.Swap(randomIndex, i);
5   }

是最佳的洗牌算法

Inside_Out算法

C++ stl中random_shuffle使用的就是这种算法

原理

在[0, i]之间随机一个下标j,然后用位置j的元素替换掉位置i的数字

通过54次生成的随机数取1/1,1/2,…1/54能等概率地生成这54!种结果中的一种

复杂度

空间O(1),时间O(n)

代码实现

1 public static void Shuffle(Pukes pukes)
2   {
3       int len = pukes.pukes.Length;
4       for (int i = 0; i < len; ++i)
5       {
6           int randomIndex = Random.Range(0, i + 1);
7           pukes.Swap(i, randomIndex);
8       }
9   }

random_shuffle

关于c++ stl 的random_shuffle

它的算法原理和Knuth_Durstenfeld类似

先从所有元素中选一个与位置1的元素交换,然后再从剩下的n-1个元素中选择一个放到位置2,以此类推

参考链接

维基百科-Fisher–Yates shuffle

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/155556.html原文链接:https://javaforall.net

(0)
上一篇 2022年7月3日 下午6:46
下一篇 2022年7月3日 下午6:46


相关推荐

  • manus如何使用

    manus如何使用

    2026年3月15日
    2
  • 旋转矩阵及左右乘的意义,看这一篇就够了

    旋转矩阵及左右乘的意义,看这一篇就够了旋转矩阵及左右乘的意义 别浪费时间了 看这一篇就够了前言这些天研究旋转矩阵 被教科书和视频课绕迷糊了 可悲的是 如此简单的概念竟然没有一篇文章 至少我没搜到 能够直观解释清楚 有些作者不懂就不要胡扯好吧 一气之下 我决定自己研究 经过不懈努力 终于解决了这一可爱又可恨的概念 也希望看到这篇文章的人能够静下心来好好阅读 对你肯定有帮助 一 什么是旋转矩阵 首先我需要花点儿时间说一下旋转矩阵的意义 大家一定要清楚 旋转矩阵有两个含义 坐标变换和旋转向量 二 使用步骤 1

    2026年3月19日
    1
  • java键盘钩子_HOOK使用:全局键盘钩子

    java键盘钩子_HOOK使用:全局键盘钩子//CatchKey.cpp:DefinestheentrypointfortheDLLapplication.//#define_WIN32_WINNT0x0500//设置系统版本,可以使用底层键盘钩子#defineWM_MY_SHORTS(WM_USER+105)#include”windows.h”//全局变量LPWORDg_lpdwVir…

    2022年6月9日
    34
  • 生产环境中使用Cassandra(v0.6.1) 经验小结

    生产环境中使用Cassandra(v0.6.1) 经验小结

    2021年8月6日
    53
  • Sikuli 优缺点

    Sikuli 优缺点Sikuli 的优缺点优点 1 与平台无关可以实现任意类型的 UI GUI 测试 轻松实现跨平台测试 并可以解决 web 测试中的浏览器兼容性测试问题 2 可读性强代码可读性强 用图形表示逻辑关系更生动 形象 容易理解 便于维护 3 提供 jar 包便于构建大型的测试框架 实现各类测试工具之间的互补 4 用例设计与程序实施可分离用例设计人员可以只关注图型逻辑关系和测试检查点

    2026年3月18日
    1
  • 当温度升高时二极管的反向饱和电流_二极管的反向饱和电流在20度时是

    当温度升高时二极管的反向饱和电流_二极管的反向饱和电流在20度时是三极管参数:型号8050;电流放大倍数β\betaβ=311。如下的电路。施加工作电压5V,电路中静态工作点电压如下图所示。使用TH2821ALCR表,去掉电阻R3R_3R3​之后,测量电路在加电之后的输入电阻:RiR_iRi​=3090Ω\OmegaΩ。…

    2025年10月22日
    4

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号