【用unity实现100个游戏之16】Unity程序化生成随机2D地牢游戏1(附项目源码)

文章目录

  • 先看看最终效果
  • 前言
  • 随机游走算法
  • 使用随机游走算法
  • 添加地板瓦片
    • 1. 新增TilemapVisualizer,用于可视化地图
    • 2. 瓦片素材
  • 不运行执行程序化生成地牢方法
    • 1. 先简单重构代码
    • 2. 新增Editor脚本RandomDungeonGeneratorEditor
  • 将参数保存到可编辑脚本对象(ScriptableObject)
    • 1. 定义简单随机行走数据的 ScriptableObject 类
    • 2. 配置不同的地形参数
    • 3. 调用
    • 4. 效果
  • 生成墙壁
  • 补充
  • 源码
  • 完结

先看看最终效果

在这里插入图片描述

前言

关于使用TileMap生成随机2D地图,其实之前已经有做过类似的,感兴趣可以看看:
【unity实战】随机地下城生成
【unity小技巧】Unity2D TileMap+柏林噪声生成随机地图

但是随着学习深入,发现之前做的比较粗糙和不够全面,最近又在外网看到一个程序化生成2D地牢的视频,觉得不错,所以写了这一篇学习笔记,记录分享一下。

本项目可能比较长,会分几期来讲,感兴趣的可以关注一下,方便获取后续内容,本期主要是使用随机游走算法生成随机的地牢房间。

随机游走算法

新增

/// <summary>
/// 静态类,包含用于二维环境的程序化生成算法。
/// </summary>
public static class ProceduralGenerationAlgorithms
{/// <summary>/// 在二维空间中生成简单的随机行走路径。/// </summary>/// <param name="startPosition">行走的起始位置。</param>/// <param name="walkLength">行走步数。</param>/// <returns>表示所走路径的 Vector2Int 的 HashSet。</returns>public static HashSet<Vector2Int> SimpleRandomWalk(Vector2Int startPosition, int walkLength){// 创建路径 HashSetHashSet<Vector2Int> path = new HashSet<Vector2Int>();// 将起始位置添加到路径 HashSet 中path.Add(startPosition);// 将当前位置设置为起始位置var previousPosition = startPosition;// 沿着随机方向移动,将每个新位置添加到路径 HashSet 中for (int i = 0; i < walkLength; i++){var newPosition = previousPosition + Direction2D.GetRandomCardinalDirection();path.Add(newPosition);previousPosition = newPosition;}// 返回路径 HashSetreturn path;}
}/// <summary>
/// 静态类,包含二维方向性工具。
/// </summary>
public static class Direction2D
{/// <summary>/// 二维空间中基本方向的列表。/// </summary>public static List<Vector2Int> cardinalDirectionsList = new List<Vector2Int>{new Vector2Int(0,1), //上new Vector2Int(1,0), //右new Vector2Int(0, -1), // 下new Vector2Int(-1, 0) //左};/// <summary>/// 从列表中返回一个随机的基本方向。/// </summary>/// <returns>表示随机基本方向的 Vector2Int。</returns>public static Vector2Int GetRandomCardinalDirection(){return cardinalDirectionsList[UnityEngine.Random.Range(0, cardinalDirectionsList.Count)];}
}

使用随机游走算法

新增SimpleRandomWalkDungeonGenerator, 用于生成简单随机行走地牢的类

/// <summary>
/// 用于生成简单随机行走地牢的类,继承自 MonoBehaviour。
/// </summary>
public class SimpleRandomWalkDungeonGenerator : MonoBehaviour
{[SerializeField, Header("地牢生成的起始位置")]protected Vector2Int startPosition = Vector2Int.zero;[SerializeField, Header("迭代次数")]private int iterations = 10;[SerializeField, Header("每次行走的步数")]public int walkLength = 10;[SerializeField, Header("每次迭代是否随机起始位置")]public bool startRandomlyEachIteration = true;/// <summary>/// 执行程序化生成地牢的方法。/// </summary>public void RunProceduralGeneration(){HashSet<Vector2Int> floorPositions = RunRandomWalk(); // 获取地牢地板坐标集合foreach (var position in floorPositions){Debug.Log(position); // 输出地板坐标信息}}/// <summary>/// 执行随机行走算法生成地牢地板坐标的方法。/// </summary>/// <returns>地牢地板坐标的 HashSet。</returns>protected HashSet<Vector2Int> RunRandomWalk(){var currentPosition = startPosition; // 当前位置初始化为起始位置HashSet<Vector2Int> floorPositions = new HashSet<Vector2Int>(); // 地板坐标的 HashSetfor (int i = 0; i < iterations; i++){var path = ProceduralGenerationAlgorithms.SimpleRandomWalk(currentPosition, walkLength); // 生成随机行走路径floorPositions.UnionWith(path); // 将路径添加到地板坐标集合中if (startRandomlyEachIteration){currentPosition = floorPositions.ElementAt(Random.Range(0, floorPositions.Count)); // 如果需要每次迭代随机起始位置,则随机选择一个已生成的位置}}return floorPositions; // 返回地板坐标集合}
}

挂载脚本
在这里插入图片描述
配置点击事件
在这里插入图片描述

效果
在这里插入图片描述

添加地板瓦片

1. 新增TilemapVisualizer,用于可视化地图

/// <summary>
/// 用于可视化地图的 TilemapVisualizer 类,继承自 MonoBehaviour。
/// </summary>
public class TilemapVisualizer : MonoBehaviour
{[SerializeField]private Tilemap floorTilemap; // 地板瓦片地图[SerializeField]private TileBase floorTile; // 地板瓦片/// <summary>/// 绘制地板瓦片的方法。/// </summary>/// <param name="floorPositions">地板位置的坐标集合。</param>public void PaintFloorTiles(IEnumerable<Vector2Int> floorPositions){PaintTiles(floorPositions, floorTilemap, floorTile);}/// <summary>/// 绘制瓦片的方法。/// </summary>/// <param name="positions">瓦片位置的坐标集合。</param>/// <param name="tilemap">瓦片地图。</param>/// <param name="tile">要绘制的瓦片。</param>private void PaintTiles(IEnumerable<Vector2Int> positions, Tilemap tilemap, TileBase tile){foreach (var position in positions){PaintSingleTile(tilemap, tile, position);}}/// <summary>/// 绘制单个瓦片的方法。/// </summary>/// <param name="tilemap">瓦片地图。</param>/// <param name="tile">要绘制的瓦片。</param>/// <param name="position">瓦片的位置坐标。</param>private void PaintSingleTile(Tilemap tilemap, TileBase tile, Vector2Int position){var tilePosition = tilemap.WorldToCell((Vector3Int)position); // 将位置坐标转换为瓦片地图上的单元格坐标tilemap.SetTile(tilePosition, tile); // 在指定位置绘制瓦片}//  清空瓦片地图public void Clear(){floorTilemap.ClearAllTiles();}
}

修改SimpleRandomWalkDungeonGenerator,执行程序化生成地牢的方法

[SerializeField]
private TilemapVisualizer tilemapVisualizer;/// <summary>
/// 执行程序化生成地牢的方法。
/// </summary>
public void RunProceduralGeneration()
{HashSet<Vector2Int> floorPositions = RunRandomWalk(); // 获取地牢地板坐标集合tilemapVisualizer.Clear();//  清空瓦片地图tilemapVisualizer.PaintFloorTiles(floorPositions);
}

2. 瓦片素材

https://pixel-poem.itch.io/dungeon-assetpuck
在这里插入图片描述

挂载脚本,配置参数
在这里插入图片描述
在这里插入图片描述

效果
在这里插入图片描述

不运行执行程序化生成地牢方法

每次测试都要运行程序再执行生成地图,非常的麻烦,我们可以实现不运行也可以执行程序化生成地牢的方法

1. 先简单重构代码

新增AbstractDungeonGenerator,定义抽象地牢生成器的基类

/// <summary>
/// 抽象地牢生成器的基类,继承自 MonoBehaviour。
/// </summary>
public abstract class AbstractDungeonGenerator : MonoBehaviour
{[SerializeField, Header("瓦片可视化器")]protected TilemapVisualizer tilemapVisualizer = null;[SerializeField, Header("地牢生成的起始位置")]protected Vector2Int startPosition = Vector2Int.zero;/// <summary>/// 生成地牢的方法。/// </summary>public void GenerateDungeon(){tilemapVisualizer.Clear(); // 清空瓦片可视化器RunProceduralGeneration(); // 执行程序化生成}/// <summary>/// 执行程序化生成地牢的抽象方法,需要在子类中实现具体逻辑。/// </summary>protected abstract void RunProceduralGeneration();
}

修改SimpleRandomWalkDungeonGenerator

public class SimpleRandomWalkDungeonGenerator : AbstractDungeonGenerator
{//。。。// [SerializeField, Header("地牢生成的起始位置")]// protected Vector2Int startPosition = Vector2Int.zero;// [SerializeField]// private TilemapVisualizer tilemapVisualizer;/// <summary>/// 执行程序化生成地牢的方法。/// </summary>protected override void RunProceduralGeneration(){HashSet<Vector2Int> floorPositions = RunRandomWalk(); // 获取地牢地板坐标集合tilemapVisualizer.Clear();tilemapVisualizer.PaintFloorTiles(floorPositions);}//。。。
}

修改点击事件
在这里插入图片描述
运行测试,一切正常
在这里插入图片描述

2. 新增Editor脚本RandomDungeonGeneratorEditor

在这里插入图片描述

/// <summary>
/// 自定义编辑器类,用于 RandomDungeonGenerator。
/// </summary>
[CustomEditor(typeof(AbstractDungeonGenerator), true)]
public class RandomDungeonGeneratorEditor : Editor
{private AbstractDungeonGenerator generator; // 地牢生成器对象private void Awake(){generator = (AbstractDungeonGenerator)target; // 获取目标对象并转换为地牢生成器类型}/// <summary>/// 在 Inspector 窗口中绘制自定义的 GUI。/// </summary>public override void OnInspectorGUI(){base.OnInspectorGUI(); // 绘制基类的默认 Inspector 界面if (GUILayout.Button("Create Dungeon")) // 创建地牢的按钮{generator.GenerateDungeon(); // 调用地牢生成器的方法生成地牢}}
}

效果
在这里插入图片描述

将参数保存到可编辑脚本对象(ScriptableObject)

1. 定义简单随机行走数据的 ScriptableObject 类

新增SimpleRandomWalkSO

/// <summary>
/// 简单随机行走数据的 ScriptableObject 类。
/// </summary>
[CreateAssetMenu(fileName = "SimpleRandomWalkParameters_", menuName = "PCG/SimpleRandomWalkData")]
public class SimpleRandomWalkSO : ScriptableObject
{[Header("迭代次数")]public int iterations = 10;[Header("每次行走的步数")]public int walkLength = 10;[Header("每次迭代是否随机起点")]public bool startRandomlyEachIteration = true;
}

2. 配置不同的地形参数

配置不同的参数
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

3. 调用

修改SimpleRandomWalkDungeonGenerator,调用前面定义的参数

[SerializeField]
private SimpleRandomWalkSO  randomWalkParameters;

4. 效果

大地牢
在这里插入图片描述
小地牢
在这里插入图片描述

在这里插入图片描述

生成墙壁

新增WallGenerator,墙体生成器的静态类

/// <summary>
/// 墙体生成器的静态类。
/// </summary>
public static class WallGenerator
{/// <summary>/// 创建墙体的方法。/// </summary>/// <param name="floorPositions">地板位置的集合</param>/// <param name="tilemapVisualizer">瓦片可视化器</param>public static void CreateWalls(HashSet<Vector2Int> floorPositions, TilemapVisualizer tilemapVisualizer){var basicWallPositions = FindWallsInDirections(floorPositions, Direction2D.cardinalDirectionsList);// 在每个墙体位置上绘制基本墙体foreach (var position in basicWallPositions){tilemapVisualizer.PaintSingleBasicWall(position);}}/// <summary>/// 在指定方向上查找墙体的方法。/// </summary>/// <param name="floorPositions">地板位置的集合</param>/// <param name="directionList">方向列表</param>/// <returns>墙体位置的集合</returns>private static HashSet<Vector2Int> FindWallsInDirections(HashSet<Vector2Int> floorPositions, List<Vector2Int> directionList){HashSet<Vector2Int> wallPositions = new HashSet<Vector2Int>();foreach (var position in floorPositions){foreach (var direction in directionList){var neighbourPosition = position + direction;// 如果邻居位置不在地板位置集合中,则认为是墙体位置if (!floorPositions.Contains(neighbourPosition)){wallPositions.Add(neighbourPosition);}}}return wallPositions;}
}

修改TilemapVisualizer

[SerializeField, Header("墙壁瓦片地图")]
private Tilemap wallTilemap;
[SerializeField, Header("墙壁瓦片")]
private TileBase wallTop;//绘制墙壁瓦片的方法
internal void PaintSingleBasicWall(Vector2Int position)
{PaintSingleTile(wallTilemap, wallTop, position);
}

修改SimpleRandomWalkDungeonGenerator

/// <summary>
/// 执行程序化生成地牢的方法。
/// </summary>
protected override void RunProceduralGeneration()
{HashSet<Vector2Int> floorPositions = RunRandomWalk(); // 获取地牢地板坐标集合tilemapVisualizer.PaintFloorTiles(floorPositions);//绘制地板瓦片WallGenerator.CreateWalls(floorPositions, tilemapVisualizer);//创建墙体
}//  清空瓦片地图
public void Clear()
{floorTilemap.ClearAllTiles();wallTilemap.ClearAllTiles();
}

配置参数
在这里插入图片描述
效果
在这里插入图片描述

补充

想要优化墙壁的显示,可以选择使用rule tile绘制墙壁内容,不懂得可以看我这篇文章,写的比较详细:【Unity小技巧】Unity2D TileMap的探究

还不懂的也可以看我后面的文章,后面会讲到

源码

源码会放在本项目最后一篇

完结

赠人玫瑰,手有余香!如果文章内容对你有所帮助,请不要吝啬你的点赞评论和关注,以便我第一时间收到反馈,你的每一次支持都是我不断创作的最大动力。当然如果你发现了文章中存在错误或者有更好的解决方法,也欢迎评论私信告诉我哦!

好了,我是向宇,https://xiangyu.blog.csdn.net

一位在小公司默默奋斗的开发者,出于兴趣爱好,于是最近才开始自习unity。如果你遇到任何问题,也欢迎你评论私信找我, 虽然有些问题我可能也不一定会,但是我会查阅各方资料,争取给出最好的建议,希望可以帮助更多想学编程的人,共勉~
在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/150708.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Java-final

【1】修饰变量&#xff1b; 1.public class Test { 2. //这是一个main方法&#xff0c;是程序的入口&#xff1a; 3. public static void main(String[] args) { 4. //第1种情况&#xff1a; 5. //final修饰一个变量&#xff0c;变量的值不可以改变&#…

【考研数学】正交变换后如果不是标准型怎么办?| 关于二次型标准化的一些思考

文章目录 引言一、回顾二次型的定义是什么&#xff1f;什么叫标准二次型&#xff1f;怎么化为标准型&#xff1f; 二、思考写在最后 引言 前阵子做了下 20 年真题&#xff0c;问题大大的&#xff0c;现在订正到矩阵的第一个大题&#xff0c;是关于二次型正交变换的。和常规不同…

当代职场人做分析,当然要用大数据分析工具

不管是从效率、分析的可用性以及灵活性来看&#xff0c;用大数据分析工具都还板上钉钉的。毕竟大数据分析工具集齐了大数据时代数据分析工具应具备的特点优势。 1、对接ERP&#xff0c;立得100BI报表 点击对接金蝶、用友ERP后&#xff0c;BI系统立即即可取数分析&#xff0c;…

控制实体小车cartographer建图

cartographer建图 跑通官方例程 下载官方bag https://storage.googleapis.com/cartographer-public-data/bags/backpack_2d/cartographer_paper_deutsches_museum.bag运行bag roslaunch cartographer_ros demo_backpack_2d.launch bag_filename:${HOME}/workspace/carto_ws…

Swift-day 2

1、数据绑定&#xff0c;改变标题 State private var zoomed: Bool false 属性包装器包装的变量self.title 单向绑定 self.$textInput 双向绑定 传的是数据结构 self.title self.textInput 赋值是String self._titletitle //绑定类型加下划线2、数据绑定&#xff0c;传递结构…

铝合金钻孔铣削去毛刺加工之高速电主轴解决方案

铝合金是一种轻质、高强度的材料&#xff0c;其出色的机械性能和良好的导电性、导热性使其在工业领域广受青睐特别是在航空、航天和汽车制造中&#xff0c;铝合金的身影更是随处可见。在铝合金加工过程中&#xff0c;高速电主轴可精准而高效地完成钻孔、铣削和去毛刺等任务&…

Python大数据之linux学习总结——day09_hive调优

hive调优 hive官方配置url: https://cwiki.apache.org/confluence/display/Hive/ConfigurationProperties hive命令和参数配置 hive参数配置的意义: 开发Hive应用/调优时&#xff0c;不可避免地需要设定Hive的参数。设定Hive的参数可以调优HQL代码的执行效率&#xff0c;或帮…

拼多多官方开放平台接口app商品详情接口获取实时商品详情数据演示

拼多多开放平台提供了一种名为“商品详情接口”的API接口&#xff0c;它允许卖家从自己的系统中快速获取商品信息&#xff0c;如商品标题、描述、价格、库存等&#xff0c;并将这些信息展示在自己的店铺中。通过该接口&#xff0c;卖家可以更好地管理自己的商品库存和销售&…

投资黄金:如何选择正确的黄金品种增加收益?

黄金一直以来都是备受投资者青睐的避险资产&#xff0c;然而&#xff0c;在庞大的黄金市场中&#xff0c;选择适合自己的黄金品种成为影响收益的关键因素。黄金投资并不只有一种方式&#xff0c;而是有很多种不同的黄金品种可以选择。每种黄金品种都有其独特的特点和风险&#…

python连接hive报错:TypeError: can‘t concat str to bytes

目录 一、完整报错 二、解决 三、 其他报错 一、完整报错 Traceback (most recent call last): File "D:/Gitlab/my_world/hive2csv.py", line 18, in <module> conn hive.Connection(hosthost, portport, usernameusername, passwordpassword, data…

【HarmonyOS开发】设备调试避坑指南

备注&#xff1a;通过开发验证&#xff0c;发现每个设备调试都会存在不小的差别&#xff0c;开发验证需要注意~ 1、预览器调试&#xff08;只能预览具有Entry修饰的文件&#xff09; 修改文件&#xff0c;预览器将自动刷新 注意&#xff1a;当我们只修改了Component 组件的文件…

【Linux】套接字编程

目录 套接字 IP PORT TCP和UDP的介绍 TCP UDP 网络字节序 转换接口 UDP服务器的编写 服务器的初始化 socket bind sockaddr 结构 服务器的运行 数据的收发 业务处理 客户端的编写 运行效果 拓展 套接字 &#x1f338;首先&#xff0c;我们先思考一个问题…

【深度学习】pytorch快速得到mobilenet_v2 pth 和onnx

在linux执行这个程序&#xff1a; import torch import torch.onnx from torchvision import transforms, models from PIL import Image import os# Load MobileNetV2 model model models.mobilenet_v2(pretrainedTrue) model.eval()# Download an example image from the P…

韦东山linux驱动开发学习【常更】

1.linux目录简单介绍 2.直接运行需要在$path路径下

windows 安裝字體Font

或者直接Copy到C:\Windows\fonts 目錄下

用三智者交易策略澳福加减仓轻松盈利,就是这么厉害

就是这么厉害&#xff0c; 用三智者交易策略&#xff0c;澳福通过加减仓就可以在交易市场中轻松盈利。各位投资者都知道三智者交易策略的两个重要的原则。当市场超过外部极限时&#xff0c;在向上分形的高点和向下分形的低点&#xff0c;就会跟随外部方向/分形点。 fpmarkets澳…

如何使用Docker部署Apache+Superset数据平台并远程访问?

大数据可视化BI分析工具Apache Superset实现公网远程访问 文章目录 大数据可视化BI分析工具Apache Superset实现公网远程访问前言1. 使用Docker部署Apache Superset1.1 第一步安装docker 、docker compose1.2 克隆superset代码到本地并使用docker compose启动 2. 安装cpolar内网…

LLM大模型 (chatgpt) 在搜索和推荐上的应用

目录 1 大模型在搜索的应用1.1 召回1.1.1 倒排索引1.1.2 倒排索引存在的问题1.1.3 大模型在搜索召回的应用 (实体倒排索引&#xff09; 1.2 排序1.2.1 大模型在搜索排序应用&#xff08;融入LLM实体排序&#xff09; 2 大模型在推荐的应用2.1 学术界关于大模型在推荐的研究2.2 …

采用Nexus搭建Maven私服

采用Nexus搭建Maven私服 1.采用docker安装 1.创建数据目录挂载的目录&#xff1a; /usr/local/springcloud_1113/nexus3/nexus-data2.查询并拉取镜像docker search nexus3docker pull sonatype/nexus33.查看拉取的镜像docker images4.创建docker容器&#xff1a;可能出现启动…

为关键信息基础设施安全助力!持安科技加入关保联盟

近日&#xff0c;中关村华安关键信息基础设施安全保护联盟发布了其新一批的会员单位&#xff0c;零信任办公安全代表企业持安科技成功加入&#xff0c;与联盟企业共同为关键信息基础设施提供各类支撑和保障。 中关村华安关键信息基础设施安全保护联盟由北京市科学技术委员会、中…