Unity 报错 之 InvalidOperationException: out of sync

本文探讨了Unity中遍历字典时出现InvalidOperationException:outofsync错误的原因及解决方法,介绍了正确的遍历和修改字典的方法,并给出了在使用协程时避免该错误的具体实践。

Unity 报错 之 InvalidOperationException: out of sync

报错原文:

InvalidOperationException: out of sync
System.Collections.Generic.Dictionary2+Enumerator[System.Int32,UnityEngine.Transform].VerifyState () (at /Users/builduser/buildslave/mono/build/mcs/class/corlib/System.Collections.Generic/Dictionary.cs:912) System.Collections.Generic.Dictionary2+Enumerator[System.Int32,UnityEngine.Transform].MoveNext () (at /Users/builduser/buildslave/mono/build/mcs/class/corlib/System.Collections.Generic/Dictionary.cs:278)
System.Collections.Generic.Dictionary`2+KeyCollection+Enumerator[System.Int32,UnityEngine.Transform].MoveNext () (at /Users/builduser/buildslave/mono/build/mcs/class/corlib/System.Collections.Generic/Dictionary.cs:1028)
BigHandCard+c__Iterator6.MoveNext () (at Assets/Scripts/Public/HandCards.cs:781)
UnityEngine.SetupCoroutine.InvokeMoveNext (IEnumerator enumerator, IntPtr returnValueAddress) (at C:/buildslave/unity/build/Runtime/Export/Coroutines.cs:17)


在网上查了下是在迭代器中直接修改引起的。c#是不允许你在迭代器中直接修改。
大概是这个意思【这是个错误示范】
	public void ShowMyCard(int[] card)
    {
        if (myCardsDic.Count > 0)
        {
            foreach (int k in myCardsDic.Keys)
            {
				   if(满足一个条件)
                   {
                        //删除或者修改此元素
	        			//myCardsDic.Remove(k);
	   			   }
            }
        }
    }
若有这样的需求,应该这样写,【大致逻辑:遍历字典存储满足条件的元素,然后操作你刚存储的元素】
    public void ShowMyCard(int[] card)
    {
        if (myCardsDic.Count > 0)
        {
            List<int> myCardsList = new List<int>();
            foreach (int k in myCardsDic.Keys)
            {
	               if(满足一个条件)
                   {
                        //保存这个元素
	                    myCardsList.Add(myCardsDic[k]);
	               }
            }
           foreach (int item in myCardsList)
            {
	   			//做你要做的事情
            }
        }
    }

而我遇到的问题,和上面的问题不太一样
我的问题:是在遍历字典的时候使用了协程,然后有其他的情况下调用了这个字典,从而导致了上面的这个错误的产生,报错代码大致如下:
    public IEnumerator ShowMyCard(int[] card)
    {
        if (myCardsDic.Count > 0)
        {
            foreach (int k in myCardsDic.Keys)
            {
                float x = myCardsDic[k].localScale.x;
                myCardsDic[k].DOScaleX(0, 0.02f).OnComplete(() =>
                {
                     myCardsDic[k].DOScaleX(x, 0.02f);
                });
                yield return new WaitForFixedUpdate();
            }
        }
    }
我遇到问题的解决办法:在使用过程中不使用协程的方式去操作,发现效果也还不错,修改代码如下:
    public void ShowMyCard(int[] card)
    {
        if (myCardsDic.Count > 0)
        {
            foreach (int k in myCardsDic.Keys)
            {
                float x = myCardsDic[k].localScale.x;
                myCardsDic[k].DOScaleX(0, 0.02f).OnComplete(() =>
                {
                     myCardsDic[k].DOScaleX(x, 0.02f);
                });
            }
        }
    }

You are modifying the dictionary while iterating over it. This is a big no-no.

您正在修改字典,同时遍历它。这是一个大大的禁忌。

评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

陈言必行

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值