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

news2024/11/17 13:22:02

文章目录

  • 先看看最终效果
  • 前言
  • 生成走廊
  • 生成房间
  • 修复死胡同
  • 增加走廊宽度
    • 获取走廊位置信息集合
    • 方法一
    • 方法二
  • 源码
  • 完结

先看看最终效果

在这里插入图片描述

前言

上期已经实现了房间的生成,本期紧跟着上期内容,生成走廊并结合上期内容生成连通的房间。

生成走廊

修改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]);//方法1
	    corridors[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。如果你遇到任何问题,也欢迎你评论私信找我, 虽然有些问题我可能也不一定会,但是我会查阅各方资料,争取给出最好的建议,希望可以帮助更多想学编程的人,共勉~
在这里插入图片描述

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

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

相关文章

【性能测试】稳定性/并发压力测试的TPS计算+5W并发场景设计...

目录&#xff1a;导读 前言一、Python编程入门到精通二、接口自动化项目实战三、Web自动化项目实战四、App自动化项目实战五、一线大厂简历六、测试开发DevOps体系七、常用自动化测试工具八、JMeter性能测试九、总结&#xff08;尾部小惊喜&#xff09; 前言 1、稳定性测试TPS…

js实现页面滚动时自动切换Sidebar标签,点击标签自动滚动页面

js实现页面滚动时自动切换Sidebar侧边导航标签&#xff0c;点击标签自动滚动页面 <van-sidebar class"sidebar" v-model"activeKey"><van-sidebar-item :title"i.title" click"onChange(i)" v-for"(i,k) in activeList&…

Android项目更新依赖和打包步骤和问题汇总

目录 1、Android 项目打包&#xff0c;32位包升级到64位包问题一&#xff1a;ERROR: Conflicting configuration : armeabi-v7a,x86-64,arm64-v8a,x86 in ndk abiFilters cannot be present when splits abi filters are set : x86,armeabi-v7a,arm64-v8a 2、Android项目依赖升…

使用DHorse发布SpringBoot项目到K8S

前言 在介绍DHorse的操作之前&#xff0c;先来介绍一下使用k8s发布应用的步骤&#xff0c;以SpringBoot应用为例进行说明。 1.首先从代码仓库下载代码&#xff0c;比如GitLab&#xff1b; 2.接着进行构建&#xff0c;比如使用Maven&#xff1b; 3.如果要使用k8s作为编排&am…

SAP gui 登录条目不让修改

今天碰到用户安装的GUI 770 版本&#xff0c;不让修改&#xff0c;也不让添加 后面再选项里面找到了

时尚灵感 | 数说故事《2024春夏时装周研究报告》发布

过去两个月&#xff0c;2024春夏时装周如火如荼&#xff0c;我们不仅看到后疫情时代多个世界级时尚地标恢复活力&#xff0c;明星大咖云集&#xff0c;也在各大品牌秀场中感受到了艺术、摩登、自然、人文的精彩碰撞。 时装周是流行的风向标&#xff0c;每一季四大时装周的举办…

实在智能出席山东省数据科学大会,构建产学研教数智创新生态

11月18日至19日&#xff0c;由山东大学、山东省科学技术协会、山东省大数据局主办的首届“山东省数据科学大会暨泰山科技论坛”在山东大学&#xff08;中心校区&#xff09;隆重召开&#xff0c;以“数聚慧能 科创未来”为主题&#xff0c;来自全国各地的高校、科研机构、企事业…

“茶叶创新:爆改营销策略,三个月狂销2300万“

我的朋友去年制作了一款白茶&#xff0c;并在短短三个月内将其销售量推到了2300万的高峰。你相信吗&#xff1f; 这位朋友并没有任何茶叶方面的经验&#xff0c;他只是一个有着冒险精神的企业家。他先找到了一家代工厂&#xff0c;帮助他把他的茶叶理念转化为现实。 当他把茶叶…

SAP ABAP ALV “报错用户接口 XXXX的状态 XXXX丢失“

会导致功能栏无法使用 原因如下这两个名字需要对的上&#xff0c;否则就会报错

Postgresql常用命令函数

1、string_agg()函数 1.1用法: string_agg(expression, delimiter)&#xff0c;参数类型(text, text) or (bytea, bytea)&#xff0c;返回类型和参数类型一致,第一个参数是字段名&#xff0c;第二个参数是样式&#xff0c;比如&#xff0c;或者#分隔。 1.2实战: SELECT * FR…

配置文件自动提示

1、引入依赖<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-configuration-processor</artifactId> </dependency> 2、修改IDEA配置

为什么 ConcurrentHashMap 中 key 不允许为null

考察目标 这是一个基础问题&#xff0c;主要考察 1 到 3 年经验的开发人员 ConcurrentHashMap 在实际应用中使用频率较高 考察这个问题的目的&#xff0c;是了解求职者的基本功。 所以为了表现更好&#xff0c;可以从 ConcurrentHashMap 的设计角度去回答。 问题解析 打开…

基于C#实现协同推荐 SlopeOne 算法

一、概念 相信大家对如下的 Category 都很熟悉&#xff0c;很多网站都有类似如下的功能&#xff0c;“商品推荐”,"猜你喜欢“&#xff0c;在实体店中我们有导购来为我们服务&#xff0c;在网络上我们需要同样的一种替代物&#xff0c;如果简简单单的在数据库里面去捞&am…

CTF-栈溢出-基本ROP-【ret2syscall】

文章目录 ret2syscallBxMCTF 2023 Anti-Libcmainwrite_bufflush_obufreadintread_buf 思路exp ret2syscall 即控制程序执行系统调用&#xff0c;获取 shell。 BxMCTF 2023 Anti-Libc main write_buf 写入字符的&#xff0c;待会输出 flush_obuf 把字符输出到屏幕 read…

集群创建(flannel)时候,没有自动创建出cni0网卡

给旧的集群加入四台新的服务器启动时候发现都是正常的&#xff0c;但是pod通信报错 集群通信失败&#xff0c;第一时刻想看看是不是cni0和flannel.1的网段是不是通的&#xff0c;点进去一看发现cni0网卡没有生成。 部署是通过kubeadm方式部署的集群&#xff0c;目前有两种解决…

mysql8 group by出错:this is incompatible with sql_mode=only_full_group_by

第一步&#xff1a;先检查自己windows电脑上有没有my.ini文件&#xff0c;没有则创建一个my.ini 第二步&#xff1a;将下面内容复制进入my.ini保存&#xff0c;重启Mysql即可 [mysqld] sql_modeSTRICT_TRANS_TABLES,NO_ZERO_IN_DATE,NO_ZERO_DATE,ERROR_FOR_DIVISION_BY_ZERO,…

请求的接口响应状态为已取消的原因

有趣的iframe问题 今天遇到一个问题&#xff0c;当点击了按钮----跳转页面时----F12键点击网络中的状态报了已取消&#xff0c;类型是 document说明是前端页面的问题&#xff0c;如果是xhr那可能是接口的问题。 原本是写了3个iframe,页面刷新的时候请求了第一个iframe,然后就…

android生成jks文件

jks文件用来校验微信支付 生成的方法&#xff1a;

论文《A recurrent latent variable model for sequential data》笔记:详解VRNN

A recurrent latent variable model for sequential data 背景 1 通过循环神经网络的序列建模 循环神经网络&#xff08;RNN&#xff09;可以接收一个可变长度的序列 x ( x 1 , x 2 , . . . , x T ) x (x_1, x_2, ..., x_T) x(x1​,x2​,...,xT​)作为输入&#xff0c;并通…

install安装路径设定

因为安装路径的前缀 CMAKE_INSTALL_PREFIX 都在外面都写了。 所以在具体的dll 在安装时&#xff0c;就写相对路径就行了。