在尝试启动多个线程时,索引超出了数组的范围
我有这个代码,它给了我一个“索引超出了数组的范围”。 我不知道为什么会发生这种情况,因为变量i
应该总是小于数组bla
的长度,因此不会导致此错误。
private void buttonDoSomething_Click(object sender, EventArgs e) { List t = new List(); string[] bla = textBoxBla.Lines; for (int i = 0; i some_thread_funmction(bla[i]))); t[i].Start(); } }
有人能告诉我如何解决这个问题,为什么会发生这种情况。 谢谢!
闭包是你的问题。
基本上,不是在创建lambda(在循环中)时抓取值,而是在需要时抓取它。 计算机速度如此之快,以至于它发生时已经不在循环中了。 值为3.这是一个例子(不要运行它):
private void buttonDoSomething_Click(object sender, EventArgs e) { List t = new List (); for (int i = 0; i < 3; i++) { t.Add(new Thread (() => Console.Write(i))); t[i].Start(); } }
想想你期望结果如何。 你想的是它吗?
现在运行它。
结果将是333
。
这是一些修改过的代码:
private void buttonDoSomething_Click(object sender, EventArgs e) { List t = new List (); string[] bla = textBoxBla.Lines; for (int i = 0; i < bla.Length; i++) { int y = i; //note the line above, that's where I make the int that the lambda has to grab t.Add(new Thread (() => some_thread_funmction(bla[y]))); //note that I don't use i there, I use y. t[i].Start(); } }
现在它会正常工作。 这个值在循环结束时超出了范围,因此lambda别无选择,只能在循环结束前接受它。 这将为您提供预期的结果,也不例外。
你所看到的是竞争条件。 你的for
循环在线程实际启动之前完成。 因此,当线程实际启动时, i
的值超出了数组的范围。
尝试复制索引值并传入副本。
private void buttonDoSomething_Click(object sender, EventArgs e) { List t = new List (); string[] bla = textBoxBla.Lines; for (int i = 0; i < bla.Length; i++) { int index = i; t.Add(new Thread (() => some_thread_funmction(bla[index]))); t[i].Start(); } }