Capturing local variables from Coroutine's for-loop woes
Unity3D/Problems 2015. 10. 6. 15:00반응형
http://forum.unity3d.com/threads/capturing-local-variables-from-coroutines-for-loop-woes.282623/
- Hey everyone,
So as I was working on some code I ran into a bug that I couldn't quite place. After some experimentation I found that the problem lies in capturing the value of a Coroutine's loop-local variable. While it works as expected when the method the binding takes place in is a "regular method" (i.e. void), it seems to always hold the same value when done from within a Coroutine. This value is always the same as the last value it held. I think a code sample to reproduce the problem is probably clearer. - Code (csharp):
- using UnityEngine;
- using System;
- using System.Collections;
- using System.Collections.Generic;
- public List<Action> actions = new List<Action>();
- yield return null;
- for(int i=0; i<3; i++){
- int_i = i;
- }
- Init();
- }
- void DoThing(int index){
- }
- void Init(){
- actions[i]();
- }
- }
- }
- The expected output here would be:
- Code (csharp):
- Adding DoThing() for index 0
- Adding DoThing() for index 1
- Adding DoThing() for index 2
- Doing the thing 0!
- Doing the thing 1!
- Doing the thing 2!
- But instead, when using Start() as a Coroutine I get:
- Code (csharp):
- Adding DoThing() for index 0
- Adding DoThing() for index 1
- Adding DoThing() for index 2
- Doing the thing 2!
- Doing the thing 2!
- Doing the thing 2!
- Is this a known limitation of Coroutines? Why is this happening? And is there an easy fix for this problem, or would I need to refactor my code so as to completely avoid this scenario?
So many questions, but looking forward to an answer! =)
Thanks,
Patrick - Stoven likes this.
Interesting!
I have an intuition as to what's going on but I'm not sure yet. Do you get the same result if you compile your code to a DLL with Microsoft's compiler, instead of the built-in Mono compiler?That's a pretty good word to describe this with!
Can you provide some steps to take to do/ test this? I've never done anything like it, and I'm not even sure if it's possible for me as I'm on OSX? If it is I'd love to give it a try though!Oof, this board is moving fast! Quoting you real quick so you get a notification. =)
- Wow, that's functionality you'd expect from Javascript in terms of variables having function scope, and not block scope. I'm not sure why it's working that way in C#? Maybe IEnumerator instances from methods are implemented differently than I had originally expected.
My guess now is that an IEnumerator method call returns an instance that has state saved for every local variable in a customized extension of IEnumerator that Unity provides.
Example of function scope in Javascript: Code (JavaScript):
var a = 1; function four() { if (true) { var a = 4; } alert(a); // alerts '4', not the global value of '1' }
- Last edited: Nov 28, 2014
Hmm, but then that would mean that (the first) int _i has its state saved as well, instead of being created every iteration? That doesn't sound like the intended behaviour then. Or did you mean something else?
- This is complete guesstimation on my part as to how Unity handles IEnumerator instances for their methods that has return type IEnumerator and of course some testing of printing the type returned to the screen.
The way your method is probably interpreted after a call to Start() by Unity
(Note: the class is not tested btw) Code (CSharp):
public class c++_iterator<T> : IEnumerator where T : struct { private int i; private int _i; private int chunk = 0; private YieldInstruction _yield; public object Current { get { return _yield; } } // ... code to set values, except chunk public bool MoveNext() { switch(chunk) { case 0: _yield = null; chunk = 1; return true; case 1: for(i=0; i<3; i++) { _i = i; } Init(); chunk = 2; return false; } return false; } { // possibly some kind of disposal for whatever types needed it } }
- Since the Anonymous method assignment from actions.Add(() => DoThing(_i)) is using the reference from the specially constructed class, and since it isn't called yet, all DoThing(_i) calls will get the same _i value once the actions execute their functions.
Again, this is purely a guess!!!
- Last edited: Nov 29, 2014
That's pretty much my guess as well
Note that this isn't something Unity does - it's something the compiler does. That's why I'm wondering whether you get the same results using a different compiler; strictly speaking it shouldn't be necessary to 'lift' the _i variable out to class scope because it's not accessed on either side of a yield, but a) maybe the compiler isn't smart enough to figure that out and b) maybe it shouldn't because it'd be even more confusing to have some variables behave this way and others not.
If you're on Mac, you won't be able to run the Microsoft compiler... you could try getting the most modern version of Mono/Xamarin Studio and building it in that. Otherwise, I'll try and remember to try this code out when I next boot into Windows. Unless @Stoven beats me to itStoven likes this.
Code (CSharp):
yield return null; int _i = 0; for(int i=0; i<3; i++){ { DoThing(_i); _i++; }); } Init(); }
- Senshi likes this.
Ah, but it is a guess that would make sense -- and as I'm typing this, approved by @superpig himself!
I'm trying to remember if I have an IDE installed on my Windows partition right now. Perhaps I can take a peek tonight, but as I'm new to this whole compiling-to-DLL thing I'd appreciate it if either of you could give a whirl as well. =)
Thanks!
Interesting; thanks for posting! Unfortunately, while this works for this exampe case, my real-life scenario is a bit more complicated. =/ I guess I could make it work like this, but it would become a bit unwieldly to maintain I'm afraid.
So far, I've only learned to use C# with Unity and nowhere else XD
I'm not 100% familiar enough with .Net to do something like what is requested, unfortunately. I spend most of my time studying the Unity documentation, discussions in the discussion section of the forums and working on my game XP
So methods that return an IEnumerator in .Net define this behavior, not specifically Unity? I was under the assumption that Unity was responsible for the custom c++_iterator<> instance... unless there are other workings happening under-the-hood with the compiler that I'm not quite understanding for this scenario.
I could have sworn that the representitive for this Unity video about Coroutines said that it was behavior specific to Unity, but upon rewatching the part I thought I remembered hearing it, he simply states (around 12:30 to 12:40) that "Calling [the] IEnumerator [function] returns an IEnumerator which is basically a pointer to the start of the function" - he doesn't explicitly state that this behavior is Unity-specific, so I'm now assuming this is a .Net thing?
Sorry if I'm simply adding confusion here. I'm very curious about this myself. Please correct me if I'm misinterpreting your statement.
Edit: Listened again. 11:04 in makes it a bit more clear (I glossed over it because I was making an attempt to find a specific part of the speech). So the C# compiler is responsible for this, as you've said! So it is a .Net thing... very interesting.
Edit 2: How did I miss this in the .Net specification???
"Technical Implementation
Although you write an iterator as a method, the compiler translates it into a nested class that is, in effect, a state machine. This class keeps track of the position of the iterator as long the For Each...Next or foreach loop in the client code continues.
To see what the compiler does, you can use the Ildasm.exe tool to view the Microsoft intermediate language code that is generated for an iterator method.
When you create an iterator for a class or struct, you do not have to implement the whole IEnumerator interface. When the compiler detects the iterator, it automatically generates the Current, MoveNext, and Dispose methods of the IEnumerator or IEnumerator<T>interface."Last edited: Nov 28, 2014Senshi likes this.
- @Stoven - this is a mono/.net thing, not a unity thing, I'll confirm that for you.
And yes, it works pretty much how Stoven described it. The method gets turned into its own anonymous class in which all the local variables of the method actually are fields of this class. That is how their states are saved, and this model isn't perfect. If you mix all sorts of anonymous syntax sugar together, things like this happen. You're mixing anonymous/lambda functions with iterator methods, and it doesn't handle it the way you expect.
If you have to have these lamdas inside the coroutine. Pull that bit of code out into its own method. The compiler instead of creating state fields for all those local variables will instead just call the method, and it will work as expected. Code (csharp):
using UnityEngine; using System.Collections.Generic; private List<System.Action> _actions = new List<System.Action>(); { yield return null; FillUpActions(); Init(); } private void FillUpActions() { for (int i = 0; i < 3; i++) { int j = i; { DoThing(j); }); } } private void DoThing(int i) { } private void Init() { { _actions[i](); } } }
Methods that return IEnumerator and use yield statements, yes. I wrote an article about how it all works a few years ago.
- Thanks everyone for the helpful links and explanations! Putting the for-loop in its own method indeed solves my issue, as does just placing the lambda in its own method only (which is more suitable for my scenario). But that got me thinking: Why not just keep going with this, and scope all the things! 0/ ? (Or in non-meme-speak: I'd rather keep this functionality local; without creating a specific top-level function for it.)
Code (csharp):
using UnityEngine; using System; using System.Collections; using System.Collections.Generic; List<System.Action> actions = new List<System.Action>(); Action<int> AddAction = new Action<int>((int index) => { }); //local method with its own scope! yield return null; for (int i = 0; i < 3; i++){ AddAction(i); //or if you want to get *really* conservative: new Action<int>((int index) => { })(i); } Init(); } private void DoThing(int i){ } private void Init(){ actions[i](); } } }
- So yeah, problem solved at least! I am still curious about the results of a Microsoft compiler compiled DLL, but reading this I would expect it to yield (pun not intended) the same results.
Anyway, thanks again to everyone! This was a fun thing to explore and I learned a bunch! =)
반응형
'Unity3D > Problems' 카테고리의 다른 글
슬라이딩 도어를 이용하는 케릭터 구현 (0) | 2015.11.11 |
---|---|
리스트를 전달하다가 값이 날라가는 경우... (0) | 2015.10.23 |
아이폰 빌드 jit , aot exceptions (linq) (0) | 2015.07.13 |
LINQ: How to perform .Max() on a property of all objects in a collection and return the object with maximum value (0) | 2015.07.10 |
Unity3D에서 iOS 빌드시 발생하는 에러 trampoline (0) | 2015.07.10 |