当前位置:   article > 正文

使用九宫格算法在unity中实现无限地图_unity web 无限地图加载

unity web 无限地图加载

地图,一个大型的游戏不可或缺的一部分,有的游戏内容比较丰富,相对来说,他的地图就会比较大,那么,把一整块地图和资源全部加载到游戏场景中显然是行不通的,一是浪费渲染的性能,二是全部加载出来也会造成游戏运行内存过大,达到一定阈值就会造成游戏闪退,甚至直接死机,所以,动态加载地图就成了一个大型的游戏不可缺少的一部分,就像市面上的一些3A大作,例如:GTA5、荒野大镖客、古墓丽影等等大型3A大作,玩法丰富,内容多样,像这样的游戏,必须使用动态加载地图,所以,我们今天就来实现一下在unity中动态加载地图。

现在一般的动态地图加载策略:

1.先把整个游戏的地图加载出来,然后使用地形分割插件:Terrain Slicing Dynamic Loading Kit v4.0(转载:Terrain Slicing Dynamic Loading Kit v4.0下载),然后再使用这个插件动态加载,插件使用方法,请看这里→unity 地形切割及动态加载插件Terrain Slicing Dynamic Loading Kit v4.0 快速使用说明_unity 地形插件_思依_xuni的博客-CSDN博客

 2.使用九宫格算法去动态加载,以玩家位置为中心检测点,周围九个格子为检测范围,找到周围九个格子,只要其他格子不在玩家的检测范围内,就把不在玩家检测范围内的格子回收起来,用对象池保存好,相同,如果玩家的检测范围内没有格子,就从对象池中拿出格子,放在对应位置(这里的格子泛指场景中的一块块地形)

以上这两种是我知道的两种,第一种插件分割地形,只适用于2018版本之前的(包括2018版本)使用,2018之后的地形分割功能使用不了,但是加载地形的那部分仍然可以使用,不受版本限制,所以,我建议使用第二种方式(因为我实现的就是第二种),那么我们来按照第二种的思路来实现无限地图。

首先,第二种方式,也就是九宫格算法,需要规定一个玩家检测区域,那么我们来规定一个区域:

 

 

 这样,一个玩家检测区域我们就创建好了。

然后获取玩家所在位置的坐标:

 下面是一个计算是否与玩家检测区域重合的方法:

 它的调用过程:

 

 把这些点存入到上面需要显示的集合中,然后就是核心的代码,综合以上图片,完全版代码如下:

  1. using System.Collections;
  2. using System.Collections.Generic;
  3. using UnityEngine;
  4. public class TerrainManager : MonoBehaviour
  5. {
  6. public GameObject player;
  7. private Vector3 playerpos;
  8. public float PlayerWH;
  9. public float TerrainWH;
  10. public GameObject prefab;
  11. //显示的地形块
  12. private Dictionary<Vector2, GameObject> showDic = new Dictionary<Vector2, GameObject>();
  13. //对象池
  14. private Queue<GameObject> pool = new Queue<GameObject>();
  15. void Start()
  16. {
  17. playerpos = player.transform.position;
  18. }
  19. // Update is called once per frame
  20. void Update()
  21. {
  22. if (playerpos != player.transform.position)
  23. {
  24. //需要显示的列表
  25. List<Vector2> showlist = new List<Vector2>();
  26. //创建玩家区域
  27. Rect playerRect = new Rect(player.transform.position.x, player.transform.position.z, PlayerWH, PlayerWH);
  28. //获取玩家所在
  29. int x = (int)(player.transform.position.x / TerrainWH);
  30. int z = (int)(player.transform.position.z / TerrainWH);
  31. showlist.Add(new Vector2(x, z));
  32. //右
  33. if (IsLap(playerRect, new Rect((x + 1) * TerrainWH, z * TerrainWH, TerrainWH, TerrainWH))) ;
  34. {
  35. showlist.Add(new Vector2(x + 1, z));
  36. }
  37. //左
  38. if (IsLap(playerRect, new Rect((x - 1) * TerrainWH, z * TerrainWH, TerrainWH, TerrainWH))) ;
  39. {
  40. showlist.Add(new Vector2(x - 1, z));
  41. }
  42. //前
  43. if (IsLap(playerRect, new Rect(x * TerrainWH, (z + 1) * TerrainWH, TerrainWH, TerrainWH))) ;
  44. {
  45. showlist.Add(new Vector2(x, z + 1));
  46. }
  47. //后
  48. if (IsLap(playerRect, new Rect(x * TerrainWH, (z - 1) * TerrainWH, TerrainWH, TerrainWH))) ;
  49. {
  50. showlist.Add(new Vector2(x, z - 1));
  51. }
  52. //右前
  53. if (IsLap(playerRect, new Rect((x + 1) * TerrainWH, (z + 1) * TerrainWH, TerrainWH, TerrainWH))) ;
  54. {
  55. showlist.Add(new Vector2(x + 1, z + 1));
  56. }
  57. //左前
  58. if (IsLap(playerRect, new Rect((x - 1) * TerrainWH, (z + 1) * TerrainWH, TerrainWH, TerrainWH))) ;
  59. {
  60. showlist.Add(new Vector2(x - 1, z + 1));
  61. }
  62. //右后
  63. if (IsLap(playerRect, new Rect((x + 1) * TerrainWH, (z - 1) * TerrainWH, TerrainWH, TerrainWH))) ;
  64. {
  65. showlist.Add(new Vector2(x + 1, z - 1));
  66. }
  67. //左后
  68. if (IsLap(playerRect, new Rect((x - 1) * TerrainWH, (z - 1) * TerrainWH, TerrainWH, TerrainWH))) ;
  69. {
  70. showlist.Add(new Vector2(x - 1, z - 1));
  71. }
  72. //需要删掉的集合
  73. List<Vector2> deslist = new List<Vector2>();
  74. //从正在显示的里面找到不需要显示的
  75. foreach (var item in showDic.Keys)
  76. {
  77. if (!showlist.Contains(item))
  78. {
  79. //隐藏并存入对象池
  80. showDic[item].SetActive(false);
  81. pool.Enqueue(showDic[item]);
  82. deslist.Add(item);
  83. }
  84. }
  85. //从字典中删除
  86. foreach (var item in deslist)
  87. {
  88. showDic.Remove(item);
  89. }
  90. //找到需要显示但没显示的
  91. foreach (var item in showlist)
  92. {
  93. if (!showDic.ContainsKey(item))
  94. {
  95. GameObject terrain;
  96. if (pool.Count>0)
  97. {
  98. terrain = pool.Dequeue();
  99. terrain.SetActive(true);
  100. }
  101. else
  102. {
  103. terrain = Instantiate(prefab);
  104. }
  105. terrain.transform.position = new Vector3(item.x * TerrainWH, 0, item.y * TerrainWH);
  106. showDic.Add(item, terrain);
  107. }
  108. }
  109. }
  110. playerpos = player.transform.position;
  111. }
  112. public bool IsLap(Rect a, Rect b)
  113. {
  114. float aMinX = a.x - a.width / 2;
  115. float aMaxX = a.x + a.width / 2;
  116. float aMinZ = a.y - a.height / 2;
  117. float aMaxZ = a.y + a.height / 2;
  118. float bMinX = b.x - b.width / 2;
  119. float bMaxX = b.x + b.width / 2;
  120. float bMinZ = b.y - b.height / 2;
  121. float bMaxZ = b.y + b.height / 2;
  122. if (aMinX < bMaxX &&
  123. bMinX < aMaxX &&
  124. aMinZ < bMaxZ &&
  125. bMinZ < aMaxZ)
  126. {
  127. return true;
  128. }
  129. else
  130. {
  131. return false;
  132. }
  133. }
  134. }

 运行效果如下:

 

 这样就可以实现无限地图的效果了。

九宫格算法详细讲解:游戏服务端开发-AOI-九宫格法解析(附代码)_九宫格aoi_食鱼酱的博客-CSDN博客

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/不正经/article/detail/97946
推荐阅读
相关标签
  

闽ICP备14008679号