2016-08-05 54 views
0

我正在學習有關C#中的多線程。這讓我很難,所以我正在做一些簡單的程序來更好地理解它。for循環內啓動任務:意外的行爲

我意識到,如果我有:

static void Main() 
    { 
     Task[] tasks = new Task[4]; 
     for (int i = 0; i < 4; i++) 
     { 
      tasks[i] = Task.Factory.StartNew(() => { 
       Console.WriteLine(i); 
      }); 
     } 
     Console.ReadKey(); 
    } 

輸出是4444,沒有0123就像我的預期。 這是爲什麼?

編輯:

薩迪克在他回答說,這種現象的原因是因爲我在結束一個循環變量。但是,如果我添加到我的代碼Thread.sleep(500);拉姆達聲明外,並在循環內,我收到那麼,爲什麼這種行爲由clousures理應引起不加入這一行的代碼發生?

在我看來,被問到的行爲的原因是其他。值複製到它,你應該得到預期的輸出 -

static void Main() 
    { 
     Task[] tasks = new Task[4]; 
     for (int i = 0; i < 4; i++) 
     { 
      tasks[i] = Task.Factory.StartNew(() => { 
       Console.WriteLine(i); 
      }); 
      Thread.Sleep(500); 
     } 
     //Now the output is:
     Console.ReadKey(); 
    } 
+0

「爲什麼這種行爲由clousures理應引起不與加入這一行的代碼中發生」 - 因爲你現在等待每次循環之間500毫秒,從而使'Task'對運行至完成捕獲的變量在循環的下一次迭代之前更新相同的變量。 –

回答

2

做一個局部變量:

static void Main() 
{ 
    Task[] tasks = new Task[4]; 
    for (int i = 0; i < 4; i++) 
    { 
     int x = i; 
     tasks[x] = Task.Factory.StartNew(() => { 
      Console.WriteLine(x); 
     }); 
    } 
    Console.ReadKey(); 
} 

的原因,如果你不明白我剛剛寫的代碼你得到意想不到的輸出是因爲i得到共享。 You are closing over a loop variable

+0

謝謝你的回答,Sadiq。但關鍵是爲什麼這種行爲而不是我認爲的行爲? –

+0

@PabloDeLuca - 請參閱我的編輯。 – Sadique

+0

請參閱我的編輯也。 –