Unity中使用Destroy删除游戏物体的大坑

以前使用Destroy删除游戏物体,删了就删了,也没有使用过DestroyImmediate来删除游戏物体

今天使用Destroy却碰到了一个坑,故顺便记录一下

 

在文章开始前,我们有必要区分一下使用Destroy和DestroyImmediate删除游戏物体有什么区别?

1   使用Destroy删除游戏物体,游戏物体并不会立即被删除,而是异步执行的,不会影响主线程的执行,说白了,就是它另外开一条道去执行了,这点很重要,接下来的例子就会有体现

2  使用DestroyImmediate删除游戏物体,游戏物体立即被删除,代码顺序执行,影响主线程的执行

 

 

首先先介绍一下问题的背景

我做的是一个习题系统,包括选择题、填空题和简答题,填空题的界面如下图:

Unity中使用Destroy删除游戏物体的大坑_第1张图片

为了方便描述,请允许我把

这一块东西叫做 ‘框’,如上面的叫  ‘框1’

使用AnswerPanel1为模板,生成填空题的  ‘空2’(‘框2’)、‘空3’('框3')、‘空4’('框4)等,为了程序方便操作,名称对应为AnswerPanel2、AnswerPanel3 、AnswerPanel4 等  。 因为需要填空的个数是不确定的,需要根据每道题的问题中需要填空的个数来动态生成  ‘框’

以点击  ‘上一题’的按钮为例进行说明,当用户点击该按钮时,保存用户输入的答案,并把这道题中的所有的  ‘框’(除了'框1')都进行删除,接着在这个界面加载上一道题的题目,并根据题目中空的个数来动态绘制 ‘框’的个数,最后,还需要把用户在这道题填了那些内容加载回来并显示回对应的 ‘框’。

总的业务逻辑描述如下图:

Unity中使用Destroy删除游戏物体的大坑_第2张图片

 

执行第一步时,使用Destroy删除 ‘框’时,发现一个很奇怪的问题,第三步动态绘制的 ‘框’的个数是正确的,但是到第四步时,显示回用户的答案时总是不正确(只有'框1'显示的答案正确,其它的‘框’需要显示的答案直接不显示)

出现这种情况的原因是: 使用Destroy删除时,是异步删除(它是另外开道执行删除),如本道题的 ‘框3’还没有删除,上一道题的 ‘框3’就已经绘制完毕,而且两个 ‘框3’是同名的,有两个同名的 ‘框3’,当执行到第四步时,通过GameObject.Find('框3名称')的方式找到 ‘框3’,这时找到的是本道题的  ‘框3’,并把用户对应的答案显示回找的的 ‘框3’,当本道题的 ‘框3’删除了,留下的是你所看到的上一道题的  ‘框3’(空的,本来该有的,却变成啥也没有)

 

解决办法,就行删除时,不让它另外开道进行删除,说白了,就是第一步就彻底删完,不删除就不执行第二步,这时DestroyImmediate就派上用场了,把使用Destroy的地方替换为DestroyImmediate

 

本以为一切都结束时,又踩了一个不该踩的坑。

我使用的方法是: 找到  ‘框’的父游戏物体,是上面第一张图 中的  AnswerPanel,然后删除AnswerPanel下的所有子物体(除了 '框1'),代码如下:

private RectTransform  _AnswerPanelTransform;

 _AnswerPanelTransform = GameObject.Find("AnswerPanel").GetComponent();

        //每次绘制前,都把上一次绘制的删除
        for (int childIndex =0; childIndex< _AnswerPanelTransform.childCount; childIndex++)
        {
            if (_AnswerPanelTransform.GetChild(childIndex).gameObject.name != "AnswerPanel1")
            {
                GameObject.DestroyImmediate(_AnswerPanelTransform.GetChild(childIndex).gameObject);
            }
           
        }

不知你有没有发现问题所在,是真的能够全部删除(除了'框1')吗?答案是非也,AnswerPanel每删除一个子物体,它的childCount值就减少1,就好像是你要删除一个List中的一个元素一样,当List中有10个对象的时候,如果RemoveAt掉了第0个,那么后面的9个对象都会向前移动。

 

解决办法是:设置childIndex的初始值为:_AnswerPanelTransform.childCount-1,代码如下:

 private RectTransform  _AnswerPanelTransform;

 _AnswerPanelTransform = GameObject.Find("AnswerPanel").GetComponent();

        //每次绘制前,都把上一次绘制的删除
        for (int childIndex = _AnswerPanelTransform.childCount-1; childIndex >=0; childIndex--)
        {
            if (_AnswerPanelTransform.GetChild(childIndex).gameObject.name != "AnswerPanel1")
            {
                GameObject.DestroyImmediate(_AnswerPanelTransform.GetChild(childIndex).gameObject);
            }
           
        }

 

好了,到此,结束完毕

 

你可能感兴趣的:(unity)