程序世界生成器生成重叠的地块 - Unity3D C#

如何解决程序世界生成器生成重叠的地块 - Unity3D C#

我在编码方面有点菜鸟,如果我遗漏了一些明显的东西,很抱歉。

我正致力于在程序上生成一个有限边界的垃圾场。我使用了 2D 统一教程(此处为 3 部分中的第 1 部分:https://www.youtube.com/watch?v=qAf9axsyijY&ab_channel=Blackthornprod)为我的代码制作了一个骨架,但我遇到了地面瓷砖生成重叠的问题。我在网上尝试了一些修复,但没有完全完成工作。

Physics.Overlap 使代码工作得更好,但仍然有规律的重叠。我怀疑重叠的瓷砖可能在完全相同的实例中产生,这使得 Overlap 命令错误地返回 0 colliders,但我不知道如何测试或避免它。

这是我的代码。它几乎是从上面的视频直接复制的。:

using System.Collections;
using System.Collections.Generic;
using UnityEngine;

public class spawnTile : MonoBehaviour
{
    public int openingDirection;
    //1 = N
    //2 = E
    //3 = S
    //4 = W

    private GroundTemplates templates;
    //Stores tiles to spawn

    private int rand;
    private bool spawned = false;
    private bool IsInBounds = false;
    public LayerMask checkBox;
    //Layermask to check if there is already ground at spawn point
    public Vector3 chunkScale;
    //size of check area for dup spawns
    void Start()
    {
        templates = GameObject.FindGameObjectWithTag("GroundTiles").GetComponent<GroundTemplates>();
        StartCoroutine(Spawn());
    }
    public IEnumerator Spawn()
    {
        Collider[] hitColliders = Physics.OverlapBox(transform.position,chunkScale,Quaternion.identity,checkBox); // makes an array of objects inside chunk with layer mask checkBox. Used for checking if ground is present
        yield return new WaitForSeconds(.2f); 
        Debug.Log("colliders: " + hitColliders.Length);
        if (spawned == false )
        {
            if (openingDirection == 1 && IsInBounds == true && hitColliders.Length == 0)
            {
                rand = Random.Range(0,templates.NorthRooms.Length);
                Instantiate(templates.NorthRooms[rand],transform.position,templates.NorthRooms[rand].transform.rotation);
            }
            else if (openingDirection == 2 && IsInBounds == true && hitColliders.Length == 0)
            {
                rand = Random.Range(0,templates.EastRooms.Length);
                Instantiate(templates.EastRooms[rand],templates.EastRooms[rand].transform.rotation);
            }
            else if (openingDirection == 3 && IsInBounds == true && hitColliders.Length == 0)
            {
                rand = Random.Range(0,templates.SouthRooms.Length);
                Instantiate(templates.SouthRooms[rand],templates.SouthRooms[rand].transform.rotation);
            }
            else if (openingDirection == 4 && IsInBounds == true && hitColliders.Length == 0)
            {
                rand = Random.Range(0,templates.WestRooms.Length);
                Instantiate(templates.WestRooms[rand],templates.WestRooms[rand].transform.rotation);
            }
            spawned = true;
           
        }
    }



    private void OnTriggerEnter(Collider other)
    {
        
             if (other.CompareTag("Spawn") && (other.GetComponent<spawnTile>().spawned == true))
        {
            Destroy(gameObject);
            Debug.Log("Destroyed");
        }
             else if (other.CompareTag("InBounds"))
        {
            IsInBounds = true;
        }
    }
}

输出的一些图片:

a generated map

two overlapping tiles

spawn point prefab that code is attached to

ground tile prefab example

我假设多次调试尝试的代码中有一些冗余,所以如果我能简化一些事情,请lmk。

解决方法

对视频的快速分析显示没有迹象表明原作者使用协程来生成新图块。

对于您生成的每个图块,您检查是否有重叠,然后等待 0.2 秒。正是在那里等待很容易让不同的瓷砖也检查相同的空间并发现没有瓷砖重叠。因此,两个或多个瓦片将看不到重叠,并且都写入相同的位置。

我是否可以建议清理代码,删除强制延迟并尝试这样做:

void Start ( )
{
    templates = GameObject.FindGameObjectWithTag ( "GroundTiles" ).GetComponent<GroundTemplates> ( );

    var hitColliders = Physics.OverlapBox ( transform.position,chunkScale,Quaternion.identity,checkBox ); // makes an array of objects inside chunk with layer mask checkBox. Used for checking if ground is present
    Debug.Log ( "colliders: " + hitColliders.Length );
    if ( !spawned )
    {
        if ( IsInBounds && hitColliders.Length == 0 )
        {
            GameObject [ ] rooms = null;
            switch ( openingDirection )
            {
                case 1: rooms = templates.NorthRooms; break;
                case 2: rooms = templates.EastRooms; break;
                case 3: rooms = templates.SouthRooms; break;
                case 4: rooms = templates.WestRooms; break;
                default: return;
            }

            var room = rooms [ UnityEngine.Random.Range ( 0,rooms.Length ) ];
            Instantiate ( room,transform.position,room.transform.rotation );
        }
        spawned = true;
    }
}
,

我在此线程中在米兰的帮助下获得了代码。使用他的代码后。我使用序列化字段查看未满足哪些条件,并发现 InBounds 检查未通过。所以我修改了检查,使它发生在 Start() 而不是 OnTriggerEnter();

我刚刚接触到这个解决方案,如果您有有用的反馈,请告诉我。

这是新代码

using System.Collections;
using System.Collections.Generic;
using UnityEngine;

public class betterTileSpawner : MonoBehaviour
{
    public int openingDirection;
    //1 = N
    //2 = E
    //3 = S
    //4 = W

    public LayerMask checkBox;
    //Layermask to check if there is already ground at spawn point
    public Vector3 chunkScale;
    //size of check area for dup spawns
    private Collider Boundarys;
    //Used for 
    private GroundTemplates templates;
    //Stores tiles to spawn

    private int rand;
    [SerializeField]
    private bool spawned = false;
    [SerializeField]
    private bool IsInBounds = false;
    [SerializeField]
    private string Helper;
    [SerializeField]
    private int NumColliders;

    IEnumerator Start()
    {
        templates = GameObject.FindGameObjectWithTag("GroundTiles").GetComponent<GroundTemplates>();
        yield return new WaitForSeconds(.01f);
        var hitColliders = Physics.OverlapBox(transform.position,checkBox); // makes an array of objects inside chunk with layer mask checkBox. Used for checking if ground is present
        Boundarys = GameObject.FindGameObjectWithTag("InBounds").GetComponent<Collider>();
        Debug.Log("colliders: " + hitColliders.Length);
        NumColliders = hitColliders.Length;
        Helper = "didn't do any ifs";


        if (Boundarys.bounds.Contains(transform.position))
            { IsInBounds = true; }

        if (!spawned)
            {  

            Helper = "Not Spawned " + NumColliders;
            Debug.Log("Not Spawned");
                if (IsInBounds && hitColliders.Length == 0)
                {
                    GameObject[] rooms = null;
                        switch (openingDirection)
                        {
                            case 1: rooms = templates.NorthRooms; break;
                            case 2: rooms = templates.EastRooms; break;
                            case 3: rooms = templates.SouthRooms; break;
                            case 4: rooms = templates.WestRooms; break;
                            default: Debug.Log("Default case"); break;
                        }

                    var room = rooms[UnityEngine.Random.Range(0,rooms.Length)];
                    Instantiate(room,room.transform.rotation);

                    Helper = "Should have spawned,wenth through code";
                    Debug.Log("Spawned");
                    spawned = true;
                }
            
            }
    }


    private void OnTriggerEnter(Collider other)
    {

        if (other.CompareTag("Spawn") && (other.GetComponent<betterTileSpawner>().spawned == true))
        {
            Destroy(gameObject);
            Debug.Log("Destroyed");
        }
        else if (other.CompareTag("Ground"))
        {
            Destroy(gameObject);
            Debug.Log("Destroyed");
        }
    }
}

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

相关推荐


使用本地python环境可以成功执行 import pandas as pd import matplotlib.pyplot as plt # 设置字体 plt.rcParams[&#39;font.sans-serif&#39;] = [&#39;SimHei&#39;] # 能正确显示负号 p
错误1:Request method ‘DELETE‘ not supported 错误还原:controller层有一个接口,访问该接口时报错:Request method ‘DELETE‘ not supported 错误原因:没有接收到前端传入的参数,修改为如下 参考 错误2:cannot r
错误1:启动docker镜像时报错:Error response from daemon: driver failed programming external connectivity on endpoint quirky_allen 解决方法:重启docker -&gt; systemctl r
错误1:private field ‘xxx‘ is never assigned 按Altʾnter快捷键,选择第2项 参考:https://blog.csdn.net/shi_hong_fei_hei/article/details/88814070 错误2:启动时报错,不能找到主启动类 #
报错如下,通过源不能下载,最后警告pip需升级版本 Requirement already satisfied: pip in c:\users\ychen\appdata\local\programs\python\python310\lib\site-packages (22.0.4) Coll
错误1:maven打包报错 错误还原:使用maven打包项目时报错如下 [ERROR] Failed to execute goal org.apache.maven.plugins:maven-resources-plugin:3.2.0:resources (default-resources)
错误1:服务调用时报错 服务消费者模块assess通过openFeign调用服务提供者模块hires 如下为服务提供者模块hires的控制层接口 @RestController @RequestMapping(&quot;/hires&quot;) public class FeignControl
错误1:运行项目后报如下错误 解决方案 报错2:Failed to execute goal org.apache.maven.plugins:maven-compiler-plugin:3.8.1:compile (default-compile) on project sb 解决方案:在pom.
参考 错误原因 过滤器或拦截器在生效时,redisTemplate还没有注入 解决方案:在注入容器时就生效 @Component //项目运行时就注入Spring容器 public class RedisBean { @Resource private RedisTemplate&lt;String
使用vite构建项目报错 C:\Users\ychen\work&gt;npm init @vitejs/app @vitejs/create-app is deprecated, use npm init vite instead C:\Users\ychen\AppData\Local\npm-
参考1 参考2 解决方案 # 点击安装源 协议选择 http:// 路径填写 mirrors.aliyun.com/centos/8.3.2011/BaseOS/x86_64/os URL类型 软件库URL 其他路径 # 版本 7 mirrors.aliyun.com/centos/7/os/x86
报错1 [root@slave1 data_mocker]# kafka-console-consumer.sh --bootstrap-server slave1:9092 --topic topic_db [2023-12-19 18:31:12,770] WARN [Consumer clie
错误1 # 重写数据 hive (edu)&gt; insert overwrite table dwd_trade_cart_add_inc &gt; select data.id, &gt; data.user_id, &gt; data.course_id, &gt; date_format(
错误1 hive (edu)&gt; insert into huanhuan values(1,&#39;haoge&#39;); Query ID = root_20240110071417_fe1517ad-3607-41f4-bdcf-d00b98ac443e Total jobs = 1
报错1:执行到如下就不执行了,没有显示Successfully registered new MBean. [root@slave1 bin]# /usr/local/software/flume-1.9.0/bin/flume-ng agent -n a1 -c /usr/local/softwa
虚拟及没有启动任何服务器查看jps会显示jps,如果没有显示任何东西 [root@slave2 ~]# jps 9647 Jps 解决方案 # 进入/tmp查看 [root@slave1 dfs]# cd /tmp [root@slave1 tmp]# ll 总用量 48 drwxr-xr-x. 2
报错1 hive&gt; show databases; OK Failed with exception java.io.IOException:java.lang.RuntimeException: Error in configuring object Time taken: 0.474 se
报错1 [root@localhost ~]# vim -bash: vim: 未找到命令 安装vim yum -y install vim* # 查看是否安装成功 [root@hadoop01 hadoop]# rpm -qa |grep vim vim-X11-7.4.629-8.el7_9.x
修改hadoop配置 vi /usr/local/software/hadoop-2.9.2/etc/hadoop/yarn-site.xml # 添加如下 &lt;configuration&gt; &lt;property&gt; &lt;name&gt;yarn.nodemanager.res