【用unity实现100个游戏之16】Unity程序化生成随机2D地牢游戏2(附项目源码)
文章目录
- 先看看最终效果
- 前言
- 生成走廊
- 生成房间
- 修复死胡同
- 增加走廊宽度
- 获取走廊位置信息集合
- 方法一
- 方法二
- 源码
- 完结
先看看最终效果

前言
上期已经实现了房间的生成,本期紧跟着上期内容,生成走廊并结合上期内容生成连通的房间。
生成走廊
修改ProceduralGenerationAlgorithms
/// <summary>
/// 随机生成走廊的方法。
/// </summary>
/// <param name="startPosition">起始位置</param>
/// <param name="corridorLength">走廊长度</param>
/// <returns>走廊位置列表</returns>
public static List<Vector2Int> RandomWalkCorridor(Vector2Int startPosition, int corridorLength)
{// 创建走廊位置列表List<Vector2Int> corridor = new List<Vector2Int>();// 随机选择一个基本方向var direction = Direction2D.GetRandomCardinalDirection();// 将当前位置设置为起始位置var currentPosition = startPosition;// 将起始位置添加到走廊位置列表中corridor.Add(currentPosition);// 沿着基本方向移动,并将每个新位置添加到走廊位置列表中for (int i = 0; i < corridorLength; i++){currentPosition += direction;corridor.Add(currentPosition);}// 返回走廊位置列表return corridor;
}
修改SimpleRandomWalkDungeonGenerator
HashSet<Vector2Int> floorPositions = RunRandomWalk(randomWalkParameters); // 获取地牢地板坐标集合protected HashSet<Vector2Int> RunRandomWalk(SimpleRandomWalkSO parameters)
{//。。。
}
新增CorridorFirstDungeonGenerator,走廊优先地牢生成器
// 走廊优先地牢生成器,继承自简单随机行走地牢生成器
public class CorridorFirstDungeonGenerator : SimpleRandomWalkDungeonGenerator
{[SerializeField, Header("走廊长度")] private int corridorLength = 14;[SerializeField, Header("走廊数量")] private int corridorCount = 5;[SerializeField, Header("房间占比")] [Range(0.1f, 1)] private float roomPercent = 0.8f;// 执行过程化生成protected override void RunProceduralGeneration(){CorridorFirstGeneration();}// 走廊优先生成方法private void CorridorFirstGeneration(){HashSet<Vector2Int> floorPositions = new HashSet<Vector2Int>(); // 地板位置集合CreateCorridors(floorPositions); // 创建走廊tilemapVisualizer.PaintFloorTiles(floorPositions); // 绘制地板瓦片WallGenerator.CreateWalls(floorPositions, tilemapVisualizer); // 创建墙壁}// 创建走廊的方法private void CreateCorridors(HashSet<Vector2Int> floorPositions){var currentPosition = startPosition; // 当前位置设为起始位置for (int i = 0; i < corridorCount; i++) // 循环生成指定数量的走廊{var corridor = ProceduralGenerationAlgorithms.RandomWalkCorridor(currentPosition, corridorLength); // 生成随机走廊currentPosition = corridor[corridor.Count - 1]; // 更新当前位置为走廊的最后一个位置floorPositions.UnionWith(corridor); // 将走廊位置添加到地板位置集合中}}
}
配置参数,

效果,创建了走廊

生成房间
所以后续只需要在走廊的末端生成房间即可连通各个房间,添加走廊长度以防止房间之间相交
修改CorridorFirstDungeonGenerator
// 走廊优先生成方法
private void CorridorFirstGeneration()
{HashSet<Vector2Int> floorPositions = new HashSet<Vector2Int>(); // 地板位置集合HashSet<Vector2Int> potentialRoomPositions = new HashSet<Vector2Int>();// 房间位置集合CreateCorridors(floorPositions, potentialRoomPositions);// 创建走廊HashSet<Vector2Int> roomPositions = CreateRooms(potentialRoomPositions);// 创建房间floorPositions.UnionWith(roomPositions);// 将房间位置添加到地板位置集合中tilemapVisualizer.PaintFloorTiles(floorPositions); // 绘制地板瓦片WallGenerator.CreateWalls(floorPositions, tilemapVisualizer); // 创建墙壁
}// 创建走廊的方法
private void CreateCorridors(HashSet<Vector2Int> floorPositions, HashSet<Vector2Int> potentialRoomPositions)
{var currentPosition = startPosition; // 当前位置设为起始位置potentialRoomPositions.Add(currentPosition);// 将当前位置添加到潜在房间位置集合中for (int i = 0; i < corridorCount; i++) // 循环生成指定数量的走廊{var corridor = ProceduralGenerationAlgorithms.RandomWalkCorridor(currentPosition, corridorLength); // 生成随机走廊currentPosition = corridor[corridor.Count - 1]; // 更新当前位置为走廊的最后一个位置potentialRoomPositions.Add(currentPosition);// 将当前位置添加到潜在房间位置集合中floorPositions.UnionWith(corridor); // 将走廊位置添加到地板位置集合中}
}// 创建房间的方法
private HashSet<Vector2Int> CreateRooms(HashSet<Vector2Int> potentialRoomPositions)
{HashSet<Vector2Int> roomPositions = new HashSet<Vector2Int>();int roomToCreateCount = Mathf.RoundToInt(potentialRoomPositions.Count * roomPercent);// 计算需要创建的房间数量// 根据潜在房间位置集合随机选择要创建的房间位置List<Vector2Int> roomsToCreate = potentialRoomPositions.OrderBy(x => Guid.NewGuid()).Take(roomToCreateCount).ToList();foreach (var roomPosition in roomsToCreate){var roomFloor = RunRandomWalk(randomWalkParameters, roomPosition);// 在选定的位置运行随机行走算法以生成房间地板roomPositions.UnionWith(roomFloor);// 将房间地板位置添加到房间位置集合中}return roomPositions;
}
修改SimpleRandomWalkDungeonGenerator
// 获取地牢地板坐标集合
HashSet<Vector2Int> floorPositions = RunRandomWalk(randomWalkParameters, startPosition);protected HashSet<Vector2Int> RunRandomWalk(SimpleRandomWalkSO parameters, Vector2Int position)
{var currentPosition = position; // 当前位置初始化为起始位置//。。。
}
配置参数,增加走廊长度

效果

修复死胡同
前面生成房间存在一些死胡同,这非常不好,我么需要修复一下
修改CorridorFirstDungeonGenerator
// 走廊优先生成方法
private void CorridorFirstGeneration()
{HashSet<Vector2Int> floorPositions = new HashSet<Vector2Int>(); // 地板位置集合HashSet<Vector2Int> potentialRoomPositions = new HashSet<Vector2Int>();// 房间位置集合CreateCorridors(floorPositions, potentialRoomPositions);// 创建走廊HashSet<Vector2Int> roomPositions = CreateRooms(potentialRoomPositions);// 创建房间List<Vector2Int> dradEnds = FindAllDeadEnds(floorPositions);CreateRoomsAtDeadEnd(dradEnds, roomPositions);floorPositions.UnionWith(roomPositions);// 将房间位置添加到地板位置集合中tilemapVisualizer.PaintFloorTiles(floorPositions); // 绘制地板瓦片WallGenerator.CreateWalls(floorPositions, tilemapVisualizer); // 创建墙壁
}// 在死胡同处创建房间的方法
private void CreateRoomsAtDeadEnd(List<Vector2Int> deadEnds, HashSet<Vector2Int> roomFloors)
{foreach (var position in deadEnds){if (roomFloors.Contains(position) == false){var room = RunRandomWalk(randomWalkParameters, position); // 在选定的位置运行随机行走算法以生成房间地板roomFloors.UnionWith(room); // 将房间地板位置添加到房间位置集合中}}
}// 查找所有死胡同的方法
private List<Vector2Int> FindAllDeadEnds(HashSet<Vector2Int> floorPositions)
{// 创建一个空的死路位置列表List<Vector2Int> deadEnds = new List<Vector2Int>();// 对于每个位置,检查其周围的位置数量foreach (var position in floorPositions){int neighboursCount = 0;// 遍历四个基本方向(上下左右),如果相邻位置在floorPositions中,则增加邻居计数foreach (var direction in Direction2D.cardinalDirectionsList){if (floorPositions.Contains(position + direction)){neighboursCount++;}}// 如果邻居计数为1,则将该位置添加到死路列表中if (neighboursCount == 1){deadEnds.Add(position);}}// 返回所有死路的位置列表return deadEnds;
}
效果,我们可以把走廊长度扩大,这样就实现了生成不同房间的功能

效果

增加走廊宽度
获取走廊位置信息集合
修改CorridorFirstDungeonGenerator
List<List<Vector2Int>> corridors = CreateCorridors(floorPositions, potentialRoomPositions);private List<List<Vector2Int>> CreateCorridors(HashSet<Vector2Int> floorPositions, HashSet<Vector2Int> potentialRoomPositions)
{var currentPosition = startPosition; // 当前位置设为起始位置potentialRoomPositions.Add(currentPosition);// 将当前位置添加到潜在房间位置集合中List<List<Vector2Int>> corridors = new List<List<Vector2Int>>(); // 声明并初始化走廊列表for (int i = 0; i < corridorCount; i++) // 循环生成指定数量的走廊{var corridor = ProceduralGenerationAlgorithms.RandomWalkCorridor(currentPosition, corridorLength); // 生成随机走廊currentPosition = corridor[corridor.Count - 1]; // 更新当前位置为走廊的最后一个位置potentialRoomPositions.Add(currentPosition);// 将当前位置添加到潜在房间位置集合中floorPositions.UnionWith(corridor); // 将走廊位置添加到地板位置集合中corridors.Add(corridor);}return corridors;
}
下面IncreaseCorridorSizeByOne() 和 IncreaseCorridorBrush3by3()两种办法都可以增加走廊宽度走廊,但它们的实现方式不同。
IncreaseCorridorSizeByOne()方法比较简单粗暴,适合用于简单的走廊扩展。
而 IncreaseCorridorBrush3by3()方法则更加考虑周围环境,生成的走廊形状更加自然。
但是,由于 IncreaseCorridorBrush3by3()方法的实现比较复杂,可能会增加代码的复杂度和运行时间,因此需要权衡使用场景。
方法一
修改CorridorFirstDungeonGenerator
// 走廊优先生成方法
private void CorridorFirstGeneration()
{//...// 对每条走廊进行遍历,增加走廊的大小并更新地板位置集合for (int i = 0; i < corridors.Count; i++){// 增加走廊大小的方法 生成3x3走廊corridors[i] = IncreaseCorridorSizeByOne(corridors[i]);//方法1// 将更新后的走廊位置添加到地板位置集合中floorPositions.UnionWith(corridors[i]);}
}public List<Vector2Int> IncreaseCorridorSizeByOne(List<Vector2Int> corridor)
{List<Vector2Int> newCorridor = new List<Vector2Int>(); // 新走廊坐标的列表Vector2Int previousDirection = Vector2Int.zero; // 上一个方向的单位向量for (int i = 1; i < corridor.Count; i++) // 遍历走廊坐标列表{Vector2Int directionFromCell = corridor[i] - corridor[i - 1]; // 获取当前坐标与上一个坐标之间的方向向量if (previousDirection != Vector2Int.zero && directionFromCell != previousDirection){// 处理转角情况for (int x = -1; x < 2; x++){for (int y = -1; y < 2; y++){newCorridor.Add(corridor[i - 1] + new Vector2Int(x, y)); // 将转角坐标添加到新走廊坐标列表中}}previousDirection = directionFromCell; // 更新上一个方向的单位向量}else{Vector2Int newCorridorTileOffset = GetDirection90From(directionFromCell); // 获取当前方向的90度偏移向量newCorridor.Add(corridor[i - 1]); // 添加当前坐标到新走廊坐标列表中newCorridor.Add(corridor[i - 1] + newCorridorTileOffset); // 添加当前坐标及偏移向量到新走廊坐标列表中}}return newCorridor; // 返回新走廊坐标的列表
}private Vector2Int GetDirection90From(Vector2Int direction)
{if (direction == Vector2Int.up){return Vector2Int.right; // 如果输入方向向量是向上的,返回向右的方向向量}if (direction == Vector2Int.right){return Vector2Int.down; // 如果输入方向向量是向右的,返回向下的方向向量}if (direction == Vector2Int.down){return Vector2Int.left; // 如果输入方向向量是向下的,返回向左的方向向量}if (direction == Vector2Int.left){return Vector2Int.up; // 如果输入方向向量是向左的,返回向上的方向向量}return Vector2Int.zero; // 如果输入方向向量不是上述情况之一,则返回零向量
}
生成效果,可以看到对于复杂走廊增加宽度的效果不是很好,有些走廊只有两格隔宽度

方法二
修改CorridorFirstDungeonGenerator
// 走廊优先生成方法
private void CorridorFirstGeneration()
{//...// 对每条走廊进行遍历,增加走廊的大小并更新地板位置集合for (int i = 0; i < corridors.Count; i++){// 增加走廊大小的方法 生成3x3走廊// corridors[i] = IncreaseCorridorSizeByOne(corridors[i]);//方法1corridors[i] = IncreaseCorridorBrush3by3(corridors[i]);//方法2// 将更新后的走廊位置添加到地板位置集合中floorPositions.UnionWith(corridors[i]);}
}public List<Vector2Int> IncreaseCorridorBrush3by3(List<Vector2Int> corridor)
{List<Vector2Int> newCorridor = new List<Vector2Int>(); // 新走廊坐标的列表for (int i = 1; i < corridor.Count; i++) // 遍历走廊坐标列表{for (int x = -1; x < 2; x++) // 在x轴方向上遍历-1到1的范围{for (int y = -1; y < 2; y++) // 在y轴方向上遍历-1到1的范围{newCorridor.Add(corridor[i - 1] + new Vector2Int(x, y)); // 将当前坐标的周围九个坐标添加到新走廊坐标列表中}}}return newCorridor; // 返回新走廊坐标的列表
}
生成效果,稳定生成3格宽度的走廊

源码
源码会放在本项目最后一篇
完结
赠人玫瑰,手有余香!如果文章内容对你有所帮助,请不要吝啬你的点赞评论和关注,以便我第一时间收到反馈,你的每一次支持都是我不断创作的最大动力。当然如果你发现了文章中存在错误或者有更好的解决方法,也欢迎评论私信告诉我哦!
好了,我是向宇,https://xiangyu.blog.csdn.net
一位在小公司默默奋斗的开发者,出于兴趣爱好,于是最近才开始自习unity。如果你遇到任何问题,也欢迎你评论私信找我, 虽然有些问题我可能也不一定会,但是我会查阅各方资料,争取给出最好的建议,希望可以帮助更多想学编程的人,共勉~

相关文章:
【用unity实现100个游戏之16】Unity程序化生成随机2D地牢游戏2(附项目源码)
文章目录 先看看最终效果前言生成走廊生成房间修复死胡同增加走廊宽度获取走廊位置信息集合方法一方法二 源码完结 先看看最终效果 前言 上期已经实现了房间的生成,本期紧跟着上期内容,生成走廊并结合上期内容生成连通的房间。 生成走廊 修改Procedur…...
潮玩宇宙大逃杀游戏开发源码说明
潮玩宇宙大逃杀游戏是一款简单而刺激的游戏。玩家在倒计时结束前从8个房间中选择一个房间并投入宝石。倒计时结束后,系统会自动生成一个敌人,然后随机挑选一个房间并清除这个房间内的人。其余7个房间内的玩家就可以按照投入比例获得被清除掉玩家的宝石。…...
UE5 操作WebSocket
插件:https://www.unrealengine.com/marketplace/zh-CN/product/websocket-client 参考:http://dascad.net/html/websocket/bp_index.html 1. 安装Plugings 2.测试websocket服务器 http://www.websocket-test.com/ 3.连接服务器 如果在Level BP里使用&a…...
Linux文件
目录 一、基本概念 二、研究进程和被打开文件的关系 (一)w方式 (二)a方式 三、认识系统接口,操作文件 (一)认识文件描述符 (二)举例 (三)…...
素短语的定义
素短语,是指至少含有一个终结符的短语,并且除自身外,不包含更小的素短语。 最左素短语是句型中最左边的素短语。...
【华为OD题库-033】经典屏保-java
题目 DVD机在视频输出时,为了保护电视显像管,在待机状态会显示"屏保动画”,如下图所示,DVD Logo在屏幕内来回运动,碰到边缘会反弹:请根据如下要求,实现屏保Logo坐标的计算算法 1、屏幕是一个800 * 600像素的矩形&…...
clang+llvm多进程gdb调试
clangllvm多进程gdb调试 前言1. 命令行gdb2. 父进程调试3. 子进程调试4. 返回父进程 前言 在学习新增llvm的优化pass时,需要跟踪clang及llvm的调用栈。然而llvm通过posix_spawn()创建了新进程,这使得gdb调试必须有一定的技巧了。 1. 命令行gdb 以下命…...
PHP反序列化简单使用
注:比较简陋,仅供参考。 编写PHP代码,实现反序列化的时候魔法函数自动调用计算器 PHP反序列化 serialize(); 将对象序列化成字符串 unserialize(); 将字符串反序列化回对象 创建类 class Stu{ public $name; public $age; public $sex; publi…...
专业课140+总分420+东南大学920专业综合考研,信息学院通信专业考研分享
专业课140总分420东南大学920专业综合考研,信息学院通信专业考研分享 我是三月开始系统考研备战,寒假先看的高数全书,奈何在家效率极其低下,才草草看了前三四章。回校后学习的比较认真,每天大概保持10个小时左右&…...
数据结构与算法编程题11
已知两个链表A和B分别表示两个集合,其元素递增排列。 请设计算法求出A与B的交集,并存放于A链表中。 a: 1, 2, 2, 4, 5, 7, 8, 9, 10 b: 1, 2, 3, 6, 7, 8 #include <iostream> using namespace std;typedef int Elemtype; #define ERROR 0; #defin…...
【LeetCode刷题】--40.组合总和II
40.组合总和II 本题详解:回溯算法 class Solution {public List<List<Integer>> combinationSum2(int[] candidates, int target) {int len candidates.length;List<List<Integer>> res new ArrayList<>();if (len 0) {return re…...
mysql面试内容点
left join和inner join的区别 1.返回不同 innerjoin只返回两个表中联结字段相等的行。left join返回包括左表中的所有记录和右表中联结字段相等的记录。 2.数量不同 inner join的数量小于等于左表和右表中的记录数量。left join的数量以左表中的记录数量相同。 3.记录属性不同…...
msvcp140.dll是什么?msvcp140.dll丢失的有哪些解决方法
在计算机使用过程中,我们经常会遇到一些错误提示,其中之一就是“msvcp140.dll丢失”。这个错误通常会导致某些应用程序无法正常运行。为了解决这个问题,我们需要采取一些措施来修复丢失的msvcp140.dll文件。本文将详细介绍5个解决msvcp140.dl…...
数字图像处理(冈萨雷斯)学习笔记
目录 一.机器视觉和计算机视觉二.图像处理基础1.什么是图像2.如何访问图像 三.图像仿射变换四.灰度变换 一.机器视觉和计算机视觉 机器视觉(Machine Vision,MV)和计算机视觉(Computer Vision,CV)的区别和联系: 机器视觉更注重广义图像信号(激光ÿ…...
MES系统管理范围及标准
一、计划管理 1.1计划分为:月度计划>周计划>日计划; 1.2MES系统一般都会直接精确到日计划(生产工单及生产指令); 1.3MES系统日计划分为三阶排产方式: 1.3.1日计划直接排到车间,由车间自行安排任务; 1.3.2日计划排到产线或设备,对应的班组长按照计划直接生产; 1.…...
vscode运行dlv报错超时
描述 点击F5运行dlv调试go代码时报错:couldnt start dlv dap: connection timeout 解决方式 在网上搜索这个报错,据说是dlv的配置问题,修改配置后还是不行。有人说是dlv和go的版本不匹配,就朝这个方向试试 go版本改为1.19之后…...
【Leetcode合集】1. 两数之和
1. 两数之和 1. 两数之和 代码仓库地址: https://github.com/slience-me/Leetcode 个人博客 :https://slienceme.xyz 给定一个整数数组 nums 和一个整数目标值 target,请你在该数组中找出 和为目标值 target 的那 两个 整数,并…...
使用Java解决快手滑块验证码
分析页面结构: 使用浏览器开发者工具分析快手滑块验证码页面的HTML和JavaScript结构,找到滑块验证的相关元素和事件。 模拟滑块滑动: 使用Java的Selenium库或其他网络爬虫工具,模拟用户在滑块上的操作。你需要模拟鼠标点击、拖动…...
瑞吉外卖Day06
1.用户地址 1.1实体类 /*** 地址簿*/ Data public class AddressBook implements Serializable {private static final long serialVersionUID 1L;private Long id;//用户idprivate Long userId;//收货人private String consignee;//手机号private String phone;//性别 0 女…...
从暗黑3D火炬之光技能系统说到-Laya非入门教学一~资源管理
我不知道那些喷Laya没有浏览器,嘲笑别人编辑器做不好,是什么水平? 首先目前国内除了WPS和飞书,就没有第三家公司能把编辑器做好。 要是一般的游戏开发者,如我,有一点点引擎代码(某项目&#x…...
[ICLR 2022]How Much Can CLIP Benefit Vision-and-Language Tasks?
论文网址:pdf 英文是纯手打的!论文原文的summarizing and paraphrasing。可能会出现难以避免的拼写错误和语法错误,若有发现欢迎评论指正!文章偏向于笔记,谨慎食用 目录 1. 心得 2. 论文逐段精读 2.1. Abstract 2…...
c#开发AI模型对话
AI模型 前面已经介绍了一般AI模型本地部署,直接调用现成的模型数据。这里主要讲述讲接口集成到我们自己的程序中使用方式。 微软提供了ML.NET来开发和使用AI模型,但是目前国内可能使用不多,至少实践例子很少看见。开发训练模型就不介绍了&am…...
什么?连接服务器也能可视化显示界面?:基于X11 Forwarding + CentOS + MobaXterm实战指南
文章目录 什么是X11?环境准备实战步骤1️⃣ 服务器端配置(CentOS)2️⃣ 客户端配置(MobaXterm)3️⃣ 验证X11 Forwarding4️⃣ 运行自定义GUI程序(Python示例)5️⃣ 成功效果
今天再学项目的过程中遇到使用gitflow模式管理代码,因此进行学习并且发布关于gitflow的一些思考 Git与GitFlow模式 我们在写代码的时候通常会进行网上保存,无论是github还是gittee,都是一种基于git去保存代码的形式,这样保存代码…...
vulnyx Blogger writeup
信息收集 arp-scan nmap 获取userFlag 上web看看 一个默认的页面,gobuster扫一下目录 可以看到扫出的目录中得到了一个有价值的目录/wordpress,说明目标所使用的cms是wordpress,访问http://192.168.43.213/wordpress/然后查看源码能看到 这…...
华为OD机考-机房布局
import java.util.*;public class DemoTest5 {public static void main(String[] args) {Scanner in new Scanner(System.in);// 注意 hasNext 和 hasNextLine 的区别while (in.hasNextLine()) { // 注意 while 处理多个 caseSystem.out.println(solve(in.nextLine()));}}priv…...
Xela矩阵三轴触觉传感器的工作原理解析与应用场景
Xela矩阵三轴触觉传感器通过先进技术模拟人类触觉感知,帮助设备实现精确的力测量与位移监测。其核心功能基于磁性三维力测量与空间位移测量,能够捕捉多维触觉信息。该传感器的设计不仅提升了触觉感知的精度,还为机器人、医疗设备和制造业的智…...
数据结构:递归的种类(Types of Recursion)
目录 尾递归(Tail Recursion) 什么是 Loop(循环)? 复杂度分析 头递归(Head Recursion) 树形递归(Tree Recursion) 线性递归(Linear Recursion)…...
